2928 lines
81 KiB
C
2928 lines
81 KiB
C
|
/*
|
||
|
** - --=playptmod v1.05+ - 8bitbubsy 2010-2013=-- -
|
||
|
** This is the native Win32 API version, no DLL needed in you
|
||
|
** production zip/rar whatever.
|
||
|
**
|
||
|
** Thanks to mukunda for learning me how to code a .MOD player
|
||
|
** some years back!
|
||
|
**
|
||
|
** Thanks to ad_/aciddose/adejr for the BLEP and LED filter
|
||
|
** routines.
|
||
|
**
|
||
|
** Note: There's a lot of weird behavior in the coding 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 if you do.
|
||
|
**
|
||
|
** HOW DO I USE THIS FILE?
|
||
|
** Make a new file called main.c, and put this on top:
|
||
|
**
|
||
|
** #include <stdio.h>
|
||
|
**
|
||
|
** void * playptmod_Create(int soundFrequency);
|
||
|
** int playptmod_Load(void *, const char *filename);
|
||
|
** void playptmod_Play(void *);
|
||
|
** void playptmod_Render(void *, signed short *, int);
|
||
|
** void playptmod_Free(void *);
|
||
|
**
|
||
|
** void main(void)
|
||
|
** {
|
||
|
** int app_running = 1;
|
||
|
**
|
||
|
** void *p = playptmod_Create(44100);
|
||
|
** playptmod_Load(p, "hello.mod");
|
||
|
** playptmod_Play(p);
|
||
|
**
|
||
|
** while (app_running)
|
||
|
** {
|
||
|
** signed short samples[1024];
|
||
|
**
|
||
|
** if (someone_pressed_a_key())
|
||
|
** {
|
||
|
** app_running = 0;
|
||
|
** }
|
||
|
**
|
||
|
** playptmod_Render(p, samples, 512);
|
||
|
** // output samples to system here
|
||
|
**
|
||
|
** // Make sure to delay a bit here
|
||
|
** }
|
||
|
**
|
||
|
** playptmod_Free(p);
|
||
|
**
|
||
|
** return 0;
|
||
|
** }
|
||
|
**
|
||
|
**
|
||
|
** You can also integrate it as a resource in the EXE,
|
||
|
** and use some Win32 API functions to copy the MOD
|
||
|
** to memory and get a pointer to it. Then you call
|
||
|
** playptmod_LoadMem instead.
|
||
|
**
|
||
|
*/
|
||
|
|
||
|
#include "playptmod.h"
|
||
|
#include "blip_buf.h"
|
||
|
|
||
|
#include <stdio.h>
|
||
|
#include <string.h> // memcpy()
|
||
|
#include <stdlib.h> // malloc(), calloc(), free()
|
||
|
#include <math.h> // floorf(), sinf()
|
||
|
|
||
|
#define HI_NYBBLE(x) ((x) >> 4)
|
||
|
#define LO_NYBBLE(x) ((x) & 0x0F)
|
||
|
#define CLAMP(x, low, high) (((x) > (high)) ? (high) : (((x) < (low)) ? (low) : (x)))
|
||
|
#define DENORMAL_OFFSET 1E-10f
|
||
|
#define PT_MIN_PERIOD 108
|
||
|
#define PT_MAX_PERIOD 907
|
||
|
#define MAX_CHANNELS 32
|
||
|
#define MOD_ROWS 64
|
||
|
#define MOD_SAMPLES 31
|
||
|
|
||
|
#ifndef true
|
||
|
#define true 1
|
||
|
#define false 0
|
||
|
#endif
|
||
|
|
||
|
#ifdef _MSC_VER
|
||
|
#define inline __forceinline
|
||
|
#endif
|
||
|
|
||
|
enum
|
||
|
{
|
||
|
FORMAT_MK, // ProTracker 1.x
|
||
|
FORMAT_MK2, // ProTracker 2.x (if tune has >64 patterns)
|
||
|
FORMAT_FLT4, // StarTrekker
|
||
|
FORMAT_FLT8,
|
||
|
FORMAT_NCHN, // FastTracker II (only 1-9 channel MODs)
|
||
|
FORMAT_NNCH, // FastTracker II (10-32 channel MODs)
|
||
|
FORMAT_16CN, // FastTracker II (16 channel MODs)
|
||
|
FORMAT_32CN, // FastTracker II (32 channel MODs)
|
||
|
FORMAT_STK, // The Ultimate SoundTracker (15 samples)
|
||
|
FORMAT_NT, // NoiseTracker 1.0
|
||
|
|
||
|
FORMAT_MTM, // MultiTracker
|
||
|
|
||
|
FORMAT_UNKNOWN,
|
||
|
|
||
|
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;
|
||
|
short period;
|
||
|
} modnote_t;
|
||
|
|
||
|
typedef struct
|
||
|
{
|
||
|
unsigned char orderCount;
|
||
|
unsigned char patternCount;
|
||
|
unsigned char rowCount;
|
||
|
unsigned char restartPos;
|
||
|
unsigned char order[128];
|
||
|
unsigned char volume[MAX_CHANNELS];
|
||
|
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 iffSize;
|
||
|
int loopStart;
|
||
|
int loopLength;
|
||
|
int length;
|
||
|
int reallength;
|
||
|
int tmpLoopStart;
|
||
|
int offset;
|
||
|
unsigned char attribute;
|
||
|
} MODULE_SAMPLE;
|
||
|
|
||
|
typedef struct
|
||
|
{
|
||
|
char patternLoopRow;
|
||
|
char patternLoopCounter;
|
||
|
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;
|
||
|
short period;
|
||
|
short tempPeriod;
|
||
|
int noNote;
|
||
|
int invertLoopOffset;
|
||
|
int offset;
|
||
|
int offsetTemp;
|
||
|
int offsetBugNotAdded;
|
||
|
} mod_channel;
|
||
|
|
||
|
typedef struct
|
||
|
{
|
||
|
char moduleLoaded;
|
||
|
char *sampleData;
|
||
|
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];
|
||
|
float high[2];
|
||
|
} Filter;
|
||
|
|
||
|
typedef struct paula_filter_coefficients
|
||
|
{
|
||
|
float LED;
|
||
|
float LEDFb;
|
||
|
float high;
|
||
|
} FilterC;
|
||
|
|
||
|
typedef struct voice_data
|
||
|
{
|
||
|
const char *newData;
|
||
|
const char *data;
|
||
|
int index;
|
||
|
int length;
|
||
|
int loopLength;
|
||
|
int loopEnd;
|
||
|
int newLength;
|
||
|
int newLoopLength;
|
||
|
int newLoopEnd;
|
||
|
int swapSampleFlag;
|
||
|
int vol;
|
||
|
int panL;
|
||
|
int panR;
|
||
|
int step;
|
||
|
int newStep;
|
||
|
float frac;
|
||
|
float rate;
|
||
|
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;
|
||
|
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;
|
||
|
short tempPeriod;
|
||
|
int tempoTimerVal;
|
||
|
char moduleLoaded;
|
||
|
char modulePlaying;
|
||
|
char useLEDFilter;
|
||
|
unsigned short soundBufferSize;
|
||
|
unsigned int soundFrequency;
|
||
|
char soundBuffers;
|
||
|
float *frequencyTable;
|
||
|
float *extendedFrequencyTable;
|
||
|
unsigned char *sinusTable;
|
||
|
int minPeriod;
|
||
|
int maxPeriod;
|
||
|
int loopCounter;
|
||
|
int sampleCounter;
|
||
|
int samplesPerTick;
|
||
|
int vBlankTiming;
|
||
|
Voice v[MAX_CHANNELS];
|
||
|
Filter filter;
|
||
|
FilterC filterC;
|
||
|
float *mixBufferL;
|
||
|
float *mixBufferR;
|
||
|
blip_t blep[MAX_CHANNELS];
|
||
|
blip_t blepVol[MAX_CHANNELS];
|
||
|
unsigned int orderPlayed[256];
|
||
|
MODULE *source;
|
||
|
} player;
|
||
|
|
||
|
static const unsigned char invertLoopSpeeds[16] =
|
||
|
{
|
||
|
0x00, 0x05, 0x06, 0x07, 0x08, 0x0A, 0x0B, 0x0D, 0x0F, 0x13, 0x16, 0x1A, 0x20, 0x2B, 0x40, 0x80
|
||
|
};
|
||
|
|
||
|
static const short rawAmigaPeriods[606] =
|
||
|
{
|
||
|
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
|
||
|
};
|
||
|
|
||
|
static short extendedRawPeriods[16 * 85 + 13];
|
||
|
|
||
|
static const short npertab[84] =
|
||
|
{
|
||
|
/* Octaves 6 -> 0 */
|
||
|
/* C C# D D# E F F# G G# A A# B */
|
||
|
0x6b0,0x650,0x5f4,0x5a0,0x54c,0x500,0x4b8,0x474,0x434,0x3f8,0x3c0,0x38a,
|
||
|
0x358,0x328,0x2fa,0x2d0,0x2a6,0x280,0x25c,0x23a,0x21a,0x1fc,0x1e0,0x1c5,
|
||
|
0x1ac,0x194,0x17d,0x168,0x153,0x140,0x12e,0x11d,0x10d,0x0fe,0x0f0,0x0e2,
|
||
|
0x0d6,0x0ca,0x0be,0x0b4,0x0aa,0x0a0,0x097,0x08f,0x087,0x07f,0x078,0x071,
|
||
|
0x06b,0x065,0x05f,0x05a,0x055,0x050,0x04b,0x047,0x043,0x03f,0x03c,0x038,
|
||
|
0x035,0x032,0x02f,0x02d,0x02a,0x028,0x025,0x023,0x021,0x01f,0x01e,0x01c,
|
||
|
0x01b,0x019,0x018,0x016,0x015,0x014,0x013,0x012,0x011,0x010,0x00f,0x00e
|
||
|
};
|
||
|
|
||
|
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 * 3.141592f) * 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, short period)
|
||
|
{
|
||
|
char l;
|
||
|
char m;
|
||
|
char h;
|
||
|
short *tablePointer;
|
||
|
|
||
|
if (p->minPeriod == PT_MIN_PERIOD)
|
||
|
{
|
||
|
l = 0;
|
||
|
h = 35;
|
||
|
|
||
|
tablePointer = (short *)&rawAmigaPeriods[finetune * 37];
|
||
|
while (h >= l)
|
||
|
{
|
||
|
m = (h + l) / 2;
|
||
|
|
||
|
if (tablePointer[m] == period)
|
||
|
break;
|
||
|
else if (tablePointer[m] > period)
|
||
|
l = m + 1;
|
||
|
else
|
||
|
h = m - 1;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
l = 0;
|
||
|
h = 83;
|
||
|
|
||
|
tablePointer = (short *)&extendedRawPeriods[finetune * 85];
|
||
|
while (h >= l)
|
||
|
{
|
||
|
m = (h + l) / 2;
|
||
|
|
||
|
if (tablePointer[m] == period)
|
||
|
break;
|
||
|
else if (tablePointer[m] > period)
|
||
|
l = m + 1;
|
||
|
else
|
||
|
h = m - 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return (m);
|
||
|
}
|
||
|
|
||
|
static void mixerSwapChSource(player *p, int ch, const char *src, int length, int loopStart, int loopLength, int step)
|
||
|
{
|
||
|
p->v[ch].swapSampleFlag = true;
|
||
|
p->v[ch].newData = src;
|
||
|
p->v[ch].newLength = length;
|
||
|
p->v[ch].newLoopLength = loopLength;
|
||
|
p->v[ch].newLoopEnd = loopLength + loopStart;
|
||
|
p->v[ch].newStep = step;
|
||
|
}
|
||
|
|
||
|
static void mixerSetChSource(player *p, int ch, const char *src, int length, int loopStart, int loopLength, int offset, int step)
|
||
|
{
|
||
|
p->v[ch].swapSampleFlag = false;
|
||
|
p->v[ch].data = src;
|
||
|
p->v[ch].length = length;
|
||
|
p->v[ch].index = offset;
|
||
|
p->v[ch].frac = 0.0f;
|
||
|
p->v[ch].loopEnd = loopStart + loopLength;
|
||
|
p->v[ch].loopLength = loopLength;
|
||
|
p->v[ch].step = step;
|
||
|
|
||
|
if (p->v[ch].index > 0)
|
||
|
{
|
||
|
if (p->v[ch].loopLength > 2)
|
||
|
{
|
||
|
if (p->v[ch].index >= p->v[ch].loopEnd)
|
||
|
p->v[ch].index = 0;
|
||
|
}
|
||
|
else if (p->v[ch].index >= p->v[ch].length)
|
||
|
{
|
||
|
p->v[ch].data = NULL;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void mixerSetChPan(player *p, int ch, int pan)
|
||
|
{
|
||
|
p->v[ch].panL = 256 - pan;
|
||
|
p->v[ch].panR = pan;
|
||
|
}
|
||
|
|
||
|
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++)
|
||
|
{
|
||
|
ptm_blip_clear(&p->blep[i]);
|
||
|
ptm_blip_clear(&p->blepVol[i]);
|
||
|
}
|
||
|
|
||
|
memset(&p->filter, 0, sizeof (p->filter));
|
||
|
|
||
|
if (p->source)
|
||
|
{
|
||
|
for (i = 0; i < MAX_CHANNELS; i++)
|
||
|
{
|
||
|
mixerSetChVol(p, i, p->source->head.volume[i]);
|
||
|
mixerSetChPan(p, i, p->source->head.pan[i]);
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
for (i = 0; i < MAX_CHANNELS; i++)
|
||
|
{
|
||
|
mixerSetChVol(p, i, 64);
|
||
|
mixerSetChPan(p, i, (i + 1) & 2 ? 160 : 96);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void mixerSetChRate(player *p, int ch, float rate)
|
||
|
{
|
||
|
p->v[ch].rate = rate;
|
||
|
}
|
||
|
|
||
|
static void outputAudio(player *p, short *target, int numSamples)
|
||
|
{
|
||
|
short *out;
|
||
|
int i;
|
||
|
int j;
|
||
|
int step;
|
||
|
short tempSample;
|
||
|
int tempVolume;
|
||
|
float L;
|
||
|
float R;
|
||
|
|
||
|
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;
|
||
|
|
||
|
if (p->v[i].data && p->v[i].rate)
|
||
|
{
|
||
|
step = p->v[i].step;
|
||
|
for (j = 0; j < numSamples;)
|
||
|
{
|
||
|
tempSample = (p->v[i].data ? (step == 2 ? (p->v[i].data[p->v[i].index] + p->v[i].data[p->v[i].index + 1] * 0x100) : p->v[i].data[p->v[i].index] * 0x100) : 0);
|
||
|
tempVolume = (p->v[i].data && !p->v[i].mute ? p->v[i].vol : 0);
|
||
|
|
||
|
while (j < numSamples && (!p->v[i].data || p->v[i].frac >= 1.0f))
|
||
|
{
|
||
|
float t_vol = 0.0f;
|
||
|
float t_smp = 0.0f;
|
||
|
signed int i_smp;
|
||
|
|
||
|
if (p->v[i].data)
|
||
|
p->v[i].frac -= 1.0f;
|
||
|
|
||
|
t_vol += ptm_blip_read_sample(&p->blepVol[i]);
|
||
|
t_smp += ptm_blip_read_sample(&p->blep[i]);
|
||
|
|
||
|
t_smp *= t_vol;
|
||
|
i_smp = (signed int)t_smp;
|
||
|
|
||
|
p->mixBufferL[j] += i_smp * p->v[i].panL;
|
||
|
p->mixBufferR[j] += i_smp * p->v[i].panR;
|
||
|
|
||
|
j++;
|
||
|
}
|
||
|
|
||
|
if (j >= numSamples) break;
|
||
|
|
||
|
if (tempSample != p->blep[i].last_value)
|
||
|
{
|
||
|
float delta = (float)(tempSample - p->blep[i].last_value);
|
||
|
p->blep[i].last_value = tempSample;
|
||
|
ptm_blip_add_delta(&p->blep[i], p->v[i].frac, delta);
|
||
|
}
|
||
|
|
||
|
if (tempVolume != p->blepVol[i].last_value)
|
||
|
{
|
||
|
float delta = (float)(tempVolume - p->blepVol[i].last_value);
|
||
|
p->blepVol[i].last_value = tempVolume;
|
||
|
ptm_blip_add_delta(&p->blepVol[i], 0, delta);
|
||
|
}
|
||
|
|
||
|
if (p->v[i].data)
|
||
|
{
|
||
|
p->v[i].index += step;
|
||
|
p->v[i].frac += p->v[i].rate;
|
||
|
|
||
|
if (p->v[i].loopLength > (2 * step))
|
||
|
{
|
||
|
if (p->v[i].index >= p->v[i].loopEnd)
|
||
|
{
|
||
|
if (p->v[i].swapSampleFlag == true)
|
||
|
{
|
||
|
p->v[i].swapSampleFlag = false;
|
||
|
|
||
|
if (p->v[i].newLoopLength <= (2 * step))
|
||
|
{
|
||
|
p->v[i].data = NULL;
|
||
|
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
p->v[i].data = p->v[i].newData;
|
||
|
p->v[i].length = p->v[i].newLength;
|
||
|
p->v[i].loopEnd = p->v[i].newLoopEnd;
|
||
|
p->v[i].loopLength = p->v[i].newLoopLength;
|
||
|
step = p->v[i].step = p->v[i].newStep;
|
||
|
|
||
|
p->v[i].index = p->v[i].loopEnd - p->v[i].loopLength;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
p->v[i].index -= p->v[i].loopLength;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else if (p->v[i].index >= p->v[i].length)
|
||
|
{
|
||
|
if (p->v[i].swapSampleFlag == true)
|
||
|
{
|
||
|
p->v[i].swapSampleFlag = false;
|
||
|
p->v[i].data = p->v[i].newData;
|
||
|
p->v[i].length = p->v[i].newLength;
|
||
|
p->v[i].loopEnd = p->v[i].newLoopEnd;
|
||
|
p->v[i].loopLength = p->v[i].newLoopLength;
|
||
|
step = p->v[i].step = p->v[i].newStep;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
p->v[i].data = NULL;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else if (p->v[i].swapSampleFlag == true)
|
||
|
{
|
||
|
p->v[i].swapSampleFlag = false;
|
||
|
p->v[i].data = p->v[i].newData;
|
||
|
p->v[i].length = p->v[i].newLength;
|
||
|
p->v[i].loopEnd = p->v[i].newLoopEnd;
|
||
|
p->v[i].loopLength = p->v[i].newLoopLength;
|
||
|
p->v[i].step = p->v[i].newStep;
|
||
|
}
|
||
|
|
||
|
if ((j < numSamples) && (p->v[i].data == NULL))
|
||
|
{
|
||
|
for (; j < numSamples; ++j)
|
||
|
{
|
||
|
int i_smp;
|
||
|
|
||
|
tempVolume = (float)p->blepVol[i].last_value;
|
||
|
tempSample = (float)p->blep[i].last_value;
|
||
|
|
||
|
tempVolume += ptm_blip_read_sample(&p->blepVol[i]);
|
||
|
tempSample += ptm_blip_read_sample(&p->blep[i]);
|
||
|
|
||
|
tempSample *= tempVolume;
|
||
|
i_smp = (signed int)tempSample;
|
||
|
|
||
|
p->mixBufferL[j] += i_smp * p->v[i].panL;
|
||
|
p->mixBufferR[j] += i_smp * p->v[i].panR;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
out = target;
|
||
|
|
||
|
{
|
||
|
float downscale;
|
||
|
|
||
|
if (p->numChans <= 4)
|
||
|
downscale = 1.0f / (96.0f * 256.0f);
|
||
|
else
|
||
|
downscale = 1.0f / (160.0f * 256.0f);
|
||
|
|
||
|
for (i = 0; i < numSamples; ++i)
|
||
|
{
|
||
|
L = p->mixBufferL[i];
|
||
|
R = p->mixBufferR[i];
|
||
|
|
||
|
if (p->useLEDFilter == true)
|
||
|
{
|
||
|
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 -= p->filter.high[0];
|
||
|
R -= p->filter.high[1];
|
||
|
|
||
|
p->filter.high[0] += (p->filterC.high * L + DENORMAL_OFFSET);
|
||
|
p->filter.high[1] += (p->filterC.high * R + DENORMAL_OFFSET);
|
||
|
|
||
|
L *= downscale;
|
||
|
R *= downscale;
|
||
|
|
||
|
L = CLAMP(L, -32768.0f, 32767.0f);
|
||
|
R = CLAMP(R, -32768.0f, 32767.0f);
|
||
|
|
||
|
if ( out )
|
||
|
{
|
||
|
*out++ = (short)(int)(L);
|
||
|
*out++ = (short)(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] -= (p->source->head.pan[i] & 8) / 8;
|
||
|
p->source->head.pan[i] = (((int)p->source->head.pan[i]) * 255) / 14;
|
||
|
p->source->head.volume[i] = 64;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
p->source->head.pan[i] = 128;
|
||
|
p->source->head.volume[i] = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < sampleCount; ++i)
|
||
|
{
|
||
|
bufseek(fmodule, 22, SEEK_CUR);
|
||
|
|
||
|
p->source->samples[i].length = bufGetDwordLittleEndian(fmodule);
|
||
|
p->source->samples[i].loopStart = bufGetDwordLittleEndian(fmodule);
|
||
|
p->source->samples[i].loopLength = bufGetDwordLittleEndian(fmodule) - p->source->samples[i].loopStart;
|
||
|
if (p->source->samples[i].loopLength < 2)
|
||
|
p->source->samples[i].loopLength = 2;
|
||
|
|
||
|
bufread(&p->source->samples[i].fineTune, 1, 1, fmodule);
|
||
|
p->source->samples[i].fineTune = p->source->samples[i].fineTune & 0x0F;
|
||
|
|
||
|
bufread(&p->source->samples[i].volume, 1, 1, fmodule);
|
||
|
bufread(&p->source->samples[i].attribute, 1, 1, fmodule);
|
||
|
|
||
|
totalSampleSize += p->source->samples[i].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] & 0xf;
|
||
|
note[k * p->source->head.channelCount + j].param = buf[2];
|
||
|
if (note[k * p->source->head.channelCount + j].command == 0xf && note[k * p->source->head.channelCount + j].param == 0x00)
|
||
|
note[k * p->source->head.channelCount + j].command = 0;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
p->source->sampleData = (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)
|
||
|
{
|
||
|
p->source->samples[i].offset = sampleOffset;
|
||
|
bufread(&p->source->sampleData[sampleOffset], 1, p->source->samples[i].length, fmodule);
|
||
|
|
||
|
if (!(p->source->samples[i].attribute & 1))
|
||
|
{
|
||
|
for (j = (int)sampleOffset; (unsigned int)j < sampleOffset + p->source->samples[i].length; ++j)
|
||
|
p->source->sampleData[(unsigned int)j] ^= 0x80;
|
||
|
}
|
||
|
|
||
|
sampleOffset += p->source->samples[i].length;
|
||
|
}
|
||
|
|
||
|
p->source->originalSampleData = (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.initBPM = 125;
|
||
|
|
||
|
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;
|
||
|
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;
|
||
|
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;
|
||
|
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;
|
||
|
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';
|
||
|
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;
|
||
|
}
|
||
|
|
||
|
p->minPeriod = 14;
|
||
|
p->maxPeriod = 1712;
|
||
|
return;
|
||
|
}
|
||
|
else if (!strncmp(buf, "16CN", 4))
|
||
|
{
|
||
|
h->format = FORMAT_16CN;
|
||
|
p->numChans = h->channelCount = 16;
|
||
|
p->minPeriod = 14;
|
||
|
p->maxPeriod = 1712;
|
||
|
return;
|
||
|
}
|
||
|
else if (!strncmp(buf, "32CN", 4))
|
||
|
{
|
||
|
h->format = FORMAT_32CN;
|
||
|
p->numChans = h->channelCount = 32;
|
||
|
p->minPeriod = 14;
|
||
|
p->maxPeriod = 1712;
|
||
|
return;
|
||
|
}
|
||
|
else if (!strncmp(buf, "N.T.", 4))
|
||
|
{
|
||
|
h->format = FORMAT_MK; // NoiseTracker 1.0, same as ProTracker v1.x (?)
|
||
|
p->numChans = h->channelCount = 4;
|
||
|
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;
|
||
|
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];
|
||
|
char iffHdrFound;
|
||
|
int i;
|
||
|
int j;
|
||
|
int pattern;
|
||
|
int row;
|
||
|
int channel;
|
||
|
int sampleOffset;
|
||
|
int mightBeSTK;
|
||
|
int lateVerSTKFlag;
|
||
|
int numSamples;
|
||
|
int tmp;
|
||
|
unsigned int 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)
|
||
|
{
|
||
|
if ((mightBeSTK == true) && (i > 14))
|
||
|
{
|
||
|
p->source->samples[i].loopLength = 2;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
bufseek(fmodule, 22, SEEK_CUR);
|
||
|
|
||
|
p->source->samples[i].length = bufGetWordBigEndian(fmodule) * 2;
|
||
|
if (p->source->samples[i].length > 9999)
|
||
|
lateVerSTKFlag = true;
|
||
|
|
||
|
bufread(&p->source->samples[i].fineTune, 1, 1, fmodule);
|
||
|
p->source->samples[i].fineTune = p->source->samples[i].fineTune & 0x0F;
|
||
|
|
||
|
bufread(&p->source->samples[i].volume, 1, 1, fmodule);
|
||
|
if (p->source->samples[i].volume > 64)
|
||
|
p->source->samples[i].volume = 64;
|
||
|
|
||
|
if (mightBeSTK == true)
|
||
|
p->source->samples[i].loopStart = bufGetWordBigEndian(fmodule);
|
||
|
else
|
||
|
p->source->samples[i].loopStart = bufGetWordBigEndian(fmodule) * 2;
|
||
|
|
||
|
p->source->samples[i].loopLength = bufGetWordBigEndian(fmodule) * 2;
|
||
|
|
||
|
if (p->source->samples[i].loopLength < 2)
|
||
|
p->source->samples[i].loopLength = 2;
|
||
|
|
||
|
if (mightBeSTK == true)
|
||
|
{
|
||
|
if (p->source->samples[i].loopLength > 2)
|
||
|
{
|
||
|
tmp = p->source->samples[i].loopStart;
|
||
|
p->source->samples[i].length -= p->source->samples[i].loopStart;
|
||
|
p->source->samples[i].loopStart = 0;
|
||
|
p->source->samples[i].tmpLoopStart = tmp;
|
||
|
}
|
||
|
|
||
|
p->source->samples[i].fineTune = 0;
|
||
|
}
|
||
|
|
||
|
p->source->samples[i].attribute = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
bufread(&p->source->head.orderCount, 1, 1, fmodule);
|
||
|
if ((p->source->head.orderCount == 0) || (p->source->head.orderCount > 128))
|
||
|
{
|
||
|
free(p->source);
|
||
|
bufclose(fmodule);
|
||
|
|
||
|
return (false);
|
||
|
}
|
||
|
|
||
|
bufread(&p->source->head.restartPos, 1, 1, fmodule);
|
||
|
if ((mightBeSTK == true) && ((p->source->head.restartPos == 0)
|
||
|
|| (p->source->head.restartPos > 220)))
|
||
|
{
|
||
|
|
||
|
free(p->source);
|
||
|
bufclose(fmodule);
|
||
|
|
||
|
return (false);
|
||
|
}
|
||
|
|
||
|
p->source->head.initBPM = 125;
|
||
|
|
||
|
if (mightBeSTK == true)
|
||
|
{
|
||
|
p->source->head.format = FORMAT_STK;
|
||
|
|
||
|
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));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
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);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
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 = (LO_NYBBLE(bytes[0]) << 8) | bytes[1];
|
||
|
if (note->period != 0) // FLT8 is 113..856 only
|
||
|
note->period = CLAMP(note->period, 113, 856);
|
||
|
|
||
|
note->sample = (bytes[0] & 0xF0) | HI_NYBBLE(bytes[2]);
|
||
|
note->command = LO_NYBBLE(bytes[2]);
|
||
|
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 = (LO_NYBBLE(bytes[0]) << 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) | HI_NYBBLE(bytes[2]);
|
||
|
note->command = LO_NYBBLE(bytes[2]);
|
||
|
note->param = bytes[3];
|
||
|
|
||
|
if (mightBeSTK == true)
|
||
|
{
|
||
|
if (lateVerSTKFlag == false)
|
||
|
{
|
||
|
if (note->command == 0x01)
|
||
|
{
|
||
|
note->command = 0x00;
|
||
|
}
|
||
|
else if (note->command == 0x02)
|
||
|
{
|
||
|
if (note->param & 0xF0)
|
||
|
{
|
||
|
note->command = 0x02;
|
||
|
note->param >>= 4;
|
||
|
}
|
||
|
else if (note->param & 0x0F)
|
||
|
{
|
||
|
note->command = 0x01;
|
||
|
note->param &= 0x0F;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (note->command == 0x0D)
|
||
|
{
|
||
|
if (note->param == 0)
|
||
|
note->command = 0x0D;
|
||
|
else
|
||
|
note->command = 0x0A;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
note++;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
tempOffset = buftell(fmodule);
|
||
|
|
||
|
sampleOffset = 0;
|
||
|
|
||
|
numSamples = (p->source->head.format == FORMAT_STK) ? 15 : 31;
|
||
|
for (i = 0; i < numSamples; ++i)
|
||
|
{
|
||
|
iffHdrFound = 0;
|
||
|
|
||
|
s = &p->source->samples[i];
|
||
|
s->offset = sampleOffset;
|
||
|
|
||
|
j = (s->length + 1) / 2 + 5 + 16;
|
||
|
|
||
|
bufread(tempSample, 1, j, fmodule);
|
||
|
|
||
|
smpDat8 = tempSample;
|
||
|
|
||
|
if (s->length > 5 && memcmp(smpDat8, "ADPCM", 5) == 0)
|
||
|
{
|
||
|
s->reallength = j;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
s->reallength = s->length;
|
||
|
bufread(tempSample + j, 1, s->length - j, fmodule);
|
||
|
}
|
||
|
|
||
|
if (s->length > 8)
|
||
|
{
|
||
|
for (j = 0; j < (s->length - 8); ++j)
|
||
|
{
|
||
|
if (memcmp(smpDat8, "8SVXVHDR", 8) == 0)
|
||
|
iffHdrFound = 1;
|
||
|
|
||
|
if (iffHdrFound)
|
||
|
{
|
||
|
if (memcmp(smpDat8, "BODY", 4) == 0)
|
||
|
{
|
||
|
s->iffSize = j + 8;
|
||
|
s->length -= s->iffSize;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
++smpDat8;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
sampleOffset += s->length;
|
||
|
p->source->head.totalSampleSize += s->length;
|
||
|
}
|
||
|
|
||
|
p->source->sampleData = (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->iffSize > 0)
|
||
|
bufseek(fmodule, s->iffSize, SEEK_CUR);
|
||
|
|
||
|
if (s->reallength < s->length)
|
||
|
{
|
||
|
const signed char * compressionTable = tempSample + 5;
|
||
|
const unsigned char * adpcmData = tempSample + 5 + 16;
|
||
|
int delta = 0;
|
||
|
bufread(tempSample, 1, s->reallength, fmodule);
|
||
|
for ( j = 0; j < s->length; ++j )
|
||
|
{
|
||
|
delta += compressionTable[ LO_NYBBLE( *adpcmData ) ];
|
||
|
p->source->sampleData[s->offset + j] = delta;
|
||
|
if ( ++j >= s->length ) break;
|
||
|
delta += compressionTable[ HI_NYBBLE( *adpcmData ) ];
|
||
|
p->source->sampleData[s->offset + j] = delta;
|
||
|
++adpcmData;
|
||
|
}
|
||
|
}
|
||
|
else if ((mightBeSTK == true) && (p->source->samples[i].loopLength > 2))
|
||
|
{
|
||
|
for (j = 0; j < p->source->samples[i].tmpLoopStart; ++j)
|
||
|
bufseek(fmodule, 1, SEEK_CUR);
|
||
|
|
||
|
bufread(&p->source->sampleData[s->offset], 1, p->source->samples[i].length - p->source->samples[i].loopStart, fmodule);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
bufread(&p->source->sampleData[s->offset], 1, p->source->samples[i].length, fmodule);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
p->source->originalSampleData = (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;
|
||
|
memset(p->source->head.volume, 64, MAX_CHANNELS);
|
||
|
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)
|
||
|
{
|
||
|
player *p = (player *)_p;
|
||
|
if (!p->moduleLoaded)
|
||
|
{
|
||
|
int i;
|
||
|
unsigned char *buffer;
|
||
|
unsigned long fileSize;
|
||
|
FILE *fileModule;
|
||
|
|
||
|
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);
|
||
|
}
|
||
|
|
||
|
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 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,
|
||
|
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)
|
||
|
{
|
||
|
char invertLoopTemp;
|
||
|
char *invertLoopData;
|
||
|
MODULE_SAMPLE *s;
|
||
|
|
||
|
if (ch->invertLoopSpeed > 0)
|
||
|
{
|
||
|
ch->invertLoopDelay += invertLoopSpeeds[ch->invertLoopSpeed];
|
||
|
if (ch->invertLoopDelay >= 128)
|
||
|
{
|
||
|
ch->invertLoopDelay = 0;
|
||
|
|
||
|
if (ch->sample != 0)
|
||
|
{
|
||
|
s = &p->source->samples[ch->sample - 1];
|
||
|
if (s->loopLength > 2)
|
||
|
{
|
||
|
ch->invertLoopOffset++;
|
||
|
if (ch->invertLoopOffset >= (s->loopStart + s->loopLength))
|
||
|
ch->invertLoopOffset = s->loopStart;
|
||
|
|
||
|
invertLoopData = &p->source->sampleData[s->offset + ch->invertLoopOffset];
|
||
|
invertLoopTemp = -1 - *invertLoopData;
|
||
|
*invertLoopData = invertLoopTemp;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
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 -= LO_NYBBLE(ch->param);
|
||
|
|
||
|
if (p->minPeriod == PT_MIN_PERIOD)
|
||
|
{
|
||
|
if (ch->period < 113)
|
||
|
ch->period = 113;
|
||
|
}
|
||
|
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 += LO_NYBBLE(ch->param);
|
||
|
|
||
|
if (p->minPeriod == PT_MIN_PERIOD)
|
||
|
{
|
||
|
if (ch->period > 856)
|
||
|
ch->period = 856;
|
||
|
}
|
||
|
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 = LO_NYBBLE(ch->param);
|
||
|
}
|
||
|
|
||
|
static void efxVibratoControl(player *p, mod_channel *ch)
|
||
|
{
|
||
|
if (p->modTick == 0)
|
||
|
ch->vibratoControl = LO_NYBBLE(ch->param);
|
||
|
}
|
||
|
|
||
|
static void efxSetFineTune(player *p, mod_channel *ch)
|
||
|
{
|
||
|
if (p->modTick == 0)
|
||
|
ch->fineTune = LO_NYBBLE(ch->param);
|
||
|
}
|
||
|
|
||
|
static void efxPatternLoop(player *p, mod_channel *ch)
|
||
|
{
|
||
|
unsigned char tempParam;
|
||
|
|
||
|
if (p->modTick == 0)
|
||
|
{
|
||
|
tempParam = LO_NYBBLE(ch->param);
|
||
|
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 = LO_NYBBLE(ch->param);
|
||
|
}
|
||
|
|
||
|
static void efxKarplusStrong(player *p, mod_channel *ch)
|
||
|
{
|
||
|
char *sampleLoopData;
|
||
|
unsigned int loopLength;
|
||
|
unsigned int loopLengthCounter;
|
||
|
MODULE_SAMPLE *s;
|
||
|
|
||
|
if (ch->sample > 0)
|
||
|
{
|
||
|
s = &p->source->samples[ch->sample - 1];
|
||
|
|
||
|
sampleLoopData = p->source->sampleData + s->offset + s->loopStart;
|
||
|
|
||
|
loopLength = s->loopLength - 2;
|
||
|
loopLengthCounter = loopLength;
|
||
|
|
||
|
while (loopLengthCounter--)
|
||
|
{
|
||
|
*sampleLoopData = (*sampleLoopData + *(sampleLoopData + 1)) / 2;
|
||
|
sampleLoopData++;
|
||
|
}
|
||
|
|
||
|
*sampleLoopData = (*sampleLoopData + *(sampleLoopData - loopLength)) / 2;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void efxRetrigNote(player *p, mod_channel *ch)
|
||
|
{
|
||
|
unsigned char retrigTick;
|
||
|
|
||
|
retrigTick = LO_NYBBLE(ch->param);
|
||
|
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 += LO_NYBBLE(ch->param);
|
||
|
|
||
|
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 -= LO_NYBBLE(ch->param);
|
||
|
|
||
|
if (ch->volume < 0)
|
||
|
ch->volume = 0;
|
||
|
|
||
|
p->tempVolume = ch->volume;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void efxNoteCut(player *p, mod_channel *ch)
|
||
|
{
|
||
|
if (p->modTick == LO_NYBBLE(ch->param))
|
||
|
{
|
||
|
ch->volume = 0;
|
||
|
p->tempVolume = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void efxNoteDelay(player *p, mod_channel *ch)
|
||
|
{
|
||
|
unsigned char delayTick;
|
||
|
|
||
|
delayTick = LO_NYBBLE(ch->param);
|
||
|
|
||
|
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 = LO_NYBBLE(ch->param) + 1;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void efxInvertLoop(player *p, mod_channel *ch)
|
||
|
{
|
||
|
if (p->modTick == 0)
|
||
|
{
|
||
|
ch->invertLoopSpeed = LO_NYBBLE(ch->param);
|
||
|
|
||
|
if (ch->invertLoopSpeed > 0)
|
||
|
processInvertLoop(p, ch);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void handleGlissando(player *p, mod_channel *ch)
|
||
|
{
|
||
|
char l;
|
||
|
char m;
|
||
|
char h;
|
||
|
|
||
|
short *tablePointer;
|
||
|
|
||
|
if (p->tempPeriod > 0)
|
||
|
{
|
||
|
if (ch->period < ch->tempPeriod)
|
||
|
{
|
||
|
ch->period += ch->glissandoSpeed;
|
||
|
|
||
|
if (ch->period > ch->tempPeriod)
|
||
|
ch->period = ch->tempPeriod;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ch->period -= ch->glissandoSpeed;
|
||
|
|
||
|
if (ch->period < ch->tempPeriod)
|
||
|
ch->period = ch->tempPeriod;
|
||
|
}
|
||
|
|
||
|
if (ch->glissandoControl != 0)
|
||
|
{
|
||
|
if (p->minPeriod == PT_MIN_PERIOD)
|
||
|
{
|
||
|
l = 0;
|
||
|
h = 35;
|
||
|
|
||
|
tablePointer = (short *)&rawAmigaPeriods[ch->fineTune * 37];
|
||
|
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;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
l = 0;
|
||
|
h = 83;
|
||
|
|
||
|
tablePointer = (short *)&extendedRawPeriods[ch->fineTune * 85];
|
||
|
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;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
p->tempPeriod = ch->period;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
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)) // PT/NT/UST/STK
|
||
|
applyVibrato = 0;
|
||
|
|
||
|
if (applyVibrato)
|
||
|
{
|
||
|
if (p->tempPeriod > 0)
|
||
|
{
|
||
|
vibratoTemp = ch->vibratoPos >> 2;
|
||
|
vibratoTemp &= 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)) // PT/NT/UST/STK
|
||
|
applyTremolo = 0;
|
||
|
|
||
|
if (applyTremolo)
|
||
|
{
|
||
|
if (p->tempVolume > 0)
|
||
|
{
|
||
|
tremoloTemp = ch->tremoloPos >> 2;
|
||
|
tremoloTemp &= 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 += (char)tremoloData;
|
||
|
if (p->tempVolume > 64)
|
||
|
p->tempVolume = 64;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
p->tempVolume -= (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)
|
||
|
{
|
||
|
char noteToAdd;
|
||
|
char l;
|
||
|
char m;
|
||
|
char h;
|
||
|
char arpeggioTick;
|
||
|
short *tablePointer;
|
||
|
|
||
|
noteToAdd = 0;
|
||
|
|
||
|
arpeggioTick = p->modTick % 3;
|
||
|
if (arpeggioTick == 0)
|
||
|
{
|
||
|
p->tempPeriod = ch->period;
|
||
|
return;
|
||
|
}
|
||
|
else if (arpeggioTick == 1)
|
||
|
{
|
||
|
noteToAdd = HI_NYBBLE(ch->param);
|
||
|
}
|
||
|
else if (arpeggioTick == 2)
|
||
|
{
|
||
|
noteToAdd = LO_NYBBLE(ch->param);
|
||
|
}
|
||
|
|
||
|
if (p->minPeriod == PT_MIN_PERIOD) // PT/NT/UST/STK
|
||
|
{
|
||
|
l = 0;
|
||
|
h = 35;
|
||
|
|
||
|
tablePointer = (short *)&rawAmigaPeriods[ch->fineTune * 37];
|
||
|
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;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
l = 0;
|
||
|
h = 83;
|
||
|
|
||
|
tablePointer = (short *)&extendedRawPeriods[ch->fineTune * 85];
|
||
|
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->tempPeriod > 0))
|
||
|
{
|
||
|
ch->period -= ch->param;
|
||
|
|
||
|
if (p->minPeriod == PT_MIN_PERIOD)
|
||
|
{
|
||
|
if (ch->period < 113)
|
||
|
ch->period = 113;
|
||
|
}
|
||
|
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->tempPeriod > 0))
|
||
|
{
|
||
|
ch->period += ch->param;
|
||
|
|
||
|
if (p->minPeriod == PT_MIN_PERIOD)
|
||
|
{
|
||
|
if (ch->period > 856)
|
||
|
ch->period = 856;
|
||
|
}
|
||
|
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)
|
||
|
{
|
||
|
if (ch->param != 0)
|
||
|
ch->glissandoSpeed = ch->param;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
handleGlissando(p, ch);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void fxVibrato(player *p, mod_channel *ch)
|
||
|
{
|
||
|
unsigned char hiNybble;
|
||
|
unsigned char loNybble;
|
||
|
|
||
|
if (p->modTick == 0)
|
||
|
{
|
||
|
hiNybble = HI_NYBBLE(ch->param);
|
||
|
loNybble = LO_NYBBLE(ch->param);
|
||
|
|
||
|
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)
|
||
|
{
|
||
|
handleGlissando(p, ch);
|
||
|
fxVolumeSlide(p, ch);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void fxVibratoVolumeSlide(player *p, mod_channel *ch)
|
||
|
{
|
||
|
if (p->modTick > 0)
|
||
|
{
|
||
|
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 = HI_NYBBLE(ch->param);
|
||
|
loNybble = LO_NYBBLE(ch->param);
|
||
|
|
||
|
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 = ch->param * 256;
|
||
|
|
||
|
ch->offset += ch->offsetTemp;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void fxVolumeSlide(player *p, mod_channel *ch)
|
||
|
{
|
||
|
unsigned char hiNybble;
|
||
|
unsigned char loNybble;
|
||
|
|
||
|
if (p->modTick > 0)
|
||
|
{
|
||
|
hiNybble = HI_NYBBLE(ch->param);
|
||
|
loNybble = LO_NYBBLE(ch->param);
|
||
|
|
||
|
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;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
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 = ((HI_NYBBLE(ch->param) * 10) + LO_NYBBLE(ch->param));
|
||
|
|
||
|
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[HI_NYBBLE(ch->param)](p, ch);
|
||
|
}
|
||
|
|
||
|
static void fxExtended_FT2(player *p, mod_channel *ch)
|
||
|
{
|
||
|
efxRoutines_FT2[HI_NYBBLE(ch->param)](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 > 0)
|
||
|
{
|
||
|
if ((ch->param < 32) || p->vBlankTiming)
|
||
|
modSetSpeed(p, ch->param);
|
||
|
else
|
||
|
modSetTempo(p, ch->param);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
/* Bit of a hack, will alert caller that song has restarted */
|
||
|
p->modOrder = p->source->head.restartPos;
|
||
|
p->PBreakPosition = 0;
|
||
|
p->PosJumpAssert = true;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void processEffects(player *p, mod_channel *ch)
|
||
|
{
|
||
|
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)
|
||
|
mixerSetChPan(p, ch->chanIndex, (int)((float)ch->param * (256.0f / 255.0f)));
|
||
|
}
|
||
|
|
||
|
static void efxPan(player *p, mod_channel *ch)
|
||
|
{
|
||
|
if (p->modTick == 0)
|
||
|
mixerSetChPan(p, ch->chanIndex, (int)((float)LO_NYBBLE(ch->param) * (256.0f / 15.0f)));
|
||
|
}
|
||
|
|
||
|
void playptmod_Stop(void *_p)
|
||
|
{
|
||
|
player *p = (player *)_p;
|
||
|
int i;
|
||
|
|
||
|
mixerCutChannels(p);
|
||
|
|
||
|
p->modulePlaying = false;
|
||
|
|
||
|
for (i = 0; i < p->source->head.channelCount; ++i)
|
||
|
{
|
||
|
p->source->channels[i].patternLoopCounter = 0;
|
||
|
p->source->channels[i].glissandoControl = 0;
|
||
|
p->source->channels[i].vibratoControl = 0;
|
||
|
p->source->channels[i].tremoloControl = 0;
|
||
|
p->source->channels[i].fineTune = 0;
|
||
|
p->source->channels[i].invertLoopSpeed = 0;
|
||
|
p->source->channels[i].period = 0;
|
||
|
p->source->channels[i].tempPeriod = 0;
|
||
|
p->source->channels[i].offsetBugNotAdded = false;
|
||
|
}
|
||
|
|
||
|
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 > 0)
|
||
|
{
|
||
|
if (ch->command == 0x0E)
|
||
|
{
|
||
|
if (HI_NYBBLE(ch->param) == 0x05)
|
||
|
ch->fineTune = LO_NYBBLE(ch->param);
|
||
|
}
|
||
|
|
||
|
tempNote = periodToNote(p, 0, note->period);
|
||
|
|
||
|
ch->noNote = false;
|
||
|
ch->tempPeriod = (p->minPeriod == PT_MIN_PERIOD) ? rawAmigaPeriods[(ch->fineTune * 37) + tempNote] : extendedRawPeriods[(ch->fineTune * 85) + tempNote];
|
||
|
ch->flags |= FLAG_NOTE;
|
||
|
}
|
||
|
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->invertLoopOffset = s->loopStart;
|
||
|
|
||
|
if ((ch->command != 0x03) && (ch->command != 0x05))
|
||
|
{
|
||
|
ch->offset = 0;
|
||
|
ch->offsetBugNotAdded = false;
|
||
|
}
|
||
|
|
||
|
if (ch->flags & FLAG_NEWSAMPLE)
|
||
|
{
|
||
|
ch->flags &= ~FLAG_NEWSAMPLE;
|
||
|
|
||
|
if ((ch->period > 0) && ((ch->noNote == true)
|
||
|
|| (ch->command == 0x03)
|
||
|
|| (ch->command == 0x05)))
|
||
|
p->tempFlags |= TEMPFLAG_NEW_SAMPLE;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
p->tempPeriod = ch->period;
|
||
|
p->tempVolume = ch->volume;
|
||
|
|
||
|
if (p->forceEffectsOff == false)
|
||
|
processEffects(p, ch);
|
||
|
|
||
|
if (!(p->tempFlags & TEMPFLAG_DELAY))
|
||
|
{
|
||
|
if (p->tempFlags & TEMPFLAG_NEW_SAMPLE)
|
||
|
{
|
||
|
if (ch->sample > 0)
|
||
|
{
|
||
|
s = &p->source->samples[ch->sample - 1];
|
||
|
|
||
|
if (s->length > 0)
|
||
|
mixerSwapChSource(p, ch->chanIndex, p->source->sampleData + s->offset, s->length, s->loopStart, s->loopLength, s->attribute & 1 ? 2 : 1);
|
||
|
else
|
||
|
mixerSetChSource(p, ch->chanIndex, NULL, 0, 0, 0, 0, 0);
|
||
|
}
|
||
|
}
|
||
|
else 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 == false)
|
||
|
{
|
||
|
ch->offset += ch->offsetTemp;
|
||
|
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);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
mixerSetChVol(p, ch->chanIndex, p->tempVolume);
|
||
|
|
||
|
if ((p->tempPeriod >= p->minPeriod) && (p->tempPeriod <= p->maxPeriod))
|
||
|
mixerSetChRate(p, ch->chanIndex, (p->minPeriod == PT_MIN_PERIOD) ? p->frequencyTable[(int)p->tempPeriod] : p->extendedFrequencyTable[(int)p->tempPeriod]);
|
||
|
else
|
||
|
mixerSetChVol(p, ch->chanIndex, 0.0f); // arp override bugfix
|
||
|
}
|
||
|
}
|
||
|
|
||
|
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 = 0;
|
||
|
|
||
|
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;
|
||
|
|
||
|
if (p->minPeriod == PT_MIN_PERIOD) // PT/NT/STK/UST bug only
|
||
|
{
|
||
|
if (p->modTick == 0)
|
||
|
{
|
||
|
if (p->forceEffectsOff == true)
|
||
|
{
|
||
|
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);
|
||
|
|
||
|
if (p->minPeriod == PT_MIN_PERIOD) // PT/NT/STK/UST bug only
|
||
|
{
|
||
|
if (p->modTick == 0)
|
||
|
{
|
||
|
if ((p->pattBreakFlag == true) && (p->pattDelayFlag == true))
|
||
|
p->forceEffectsOff = 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 == true)
|
||
|
{
|
||
|
p->PBreakFlag = false;
|
||
|
p->modRow = p->PBreakPosition;
|
||
|
p->PBreakPosition = 0;
|
||
|
}
|
||
|
|
||
|
if ((p->modRow == p->source->head.rowCount) || (p->PosJumpAssert == true))
|
||
|
nextPosition(p);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static int pulsateSamples(player *p, int samples)
|
||
|
{
|
||
|
if (p->sampleCounter == 0)
|
||
|
{
|
||
|
processTick(p);
|
||
|
p->sampleCounter += p->samplesPerTick;
|
||
|
}
|
||
|
|
||
|
p->sampleCounter -= samples;
|
||
|
if (p->sampleCounter < 0)
|
||
|
{
|
||
|
int retSamples = samples + p->sampleCounter;
|
||
|
p->sampleCounter = 0;
|
||
|
|
||
|
return retSamples;
|
||
|
}
|
||
|
|
||
|
return samples;
|
||
|
}
|
||
|
|
||
|
void playptmod_Render(void *_p, short *target, int length)
|
||
|
{
|
||
|
player *p = (player *)_p;
|
||
|
|
||
|
if (p->modulePlaying == true)
|
||
|
{
|
||
|
static const int soundBufferSamples = soundBufferSize / 4;
|
||
|
|
||
|
while (length)
|
||
|
{
|
||
|
int tempSamples = CLAMP(length, 0, soundBufferSamples);
|
||
|
tempSamples = pulsateSamples(p, tempSamples);
|
||
|
length -= tempSamples;
|
||
|
|
||
|
outputAudio(p, target, tempSamples);
|
||
|
if ( target ) target += (tempSamples * 2);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void *playptmod_Create(int samplingFrequency)
|
||
|
{
|
||
|
player *p = (player *) calloc(1, sizeof(player));
|
||
|
|
||
|
int i, j;
|
||
|
|
||
|
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));
|
||
|
|
||
|
p->frequencyTable = (float *)malloc(sizeof (float) * 908);
|
||
|
for (i = 108; i <= 907; ++i) // 0..107 will never be looked up, junk is OK
|
||
|
p->frequencyTable[i] = (float)samplingFrequency / (7093790.0f / (2.0f * (float)i));
|
||
|
|
||
|
for (j = 0; j < 16; ++j)
|
||
|
for (i = 0; i < 85; ++i)
|
||
|
extendedRawPeriods[(j * 85) + i] = i == 84 ? 0 : npertab[i] * 8363 / finetune[j];
|
||
|
|
||
|
for (i = 0; i < 13; ++i)
|
||
|
extendedRawPeriods[16 * 85 + i] = 0;
|
||
|
|
||
|
p->soundFrequency = samplingFrequency;
|
||
|
|
||
|
p->extendedFrequencyTable = (float *)malloc(sizeof (float) * 1713);
|
||
|
for (i = 14; i <= 1712; ++i) // 0..14 will never be looked up, junk is OK
|
||
|
p->extendedFrequencyTable[i] = (float)samplingFrequency / (7093790.0f / (2.0f * (float)i));
|
||
|
|
||
|
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->filterC.high = calcRcCoeff((float)p->soundFrequency, 5.2f);
|
||
|
|
||
|
p->useLEDFilter = false;
|
||
|
|
||
|
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);
|
||
|
|
||
|
for (i = 0; i < p->source->head.channelCount; ++i)
|
||
|
{
|
||
|
p->source->channels[i].volume = 64;
|
||
|
p->source->channels[i].chanIndex = (char)i;
|
||
|
p->source->channels[i].patternLoopRow = 0;
|
||
|
p->source->channels[i].patternLoopCounter = 0;
|
||
|
p->source->channels[i].glissandoControl = 0;
|
||
|
p->source->channels[i].vibratoControl = 0;
|
||
|
p->source->channels[i].vibratoPos = 0;
|
||
|
p->source->channels[i].tremoloControl = 0;
|
||
|
p->source->channels[i].tremoloPos = 0;
|
||
|
p->source->channels[i].fineTune = 0;
|
||
|
p->source->channels[i].offsetBugNotAdded = false;
|
||
|
}
|
||
|
|
||
|
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->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->orderPlayed[startOrder] = 1;
|
||
|
p->modulePlaying = true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void playptmod_Free(void *_p)
|
||
|
{
|
||
|
player *p = (player *)_p;
|
||
|
int i;
|
||
|
|
||
|
if (p->moduleLoaded == true)
|
||
|
{
|
||
|
p->modulePlaying = false;
|
||
|
|
||
|
for (i = 0; i < 256; ++i)
|
||
|
{
|
||
|
if (p->source->patterns[i] != NULL)
|
||
|
free(p->source->patterns[i]);
|
||
|
}
|
||
|
|
||
|
free(p->source->originalSampleData);
|
||
|
free(p->source->sampleData);
|
||
|
free(p->source);
|
||
|
|
||
|
p->moduleLoaded = false;
|
||
|
}
|
||
|
|
||
|
free(p->mixBufferL);
|
||
|
free(p->mixBufferR);
|
||
|
free(p->sinusTable);
|
||
|
free(p->frequencyTable);
|
||
|
free(p->extendedFrequencyTable);
|
||
|
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.restartPos;
|
||
|
|
||
|
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 */
|