diff options
author | Jeremy Simon <jsimon13@yahoo.fr> | 2002-02-28 21:10:42 +0000 |
---|---|---|
committer | Jeremy Simon <jsimon13@yahoo.fr> | 2002-02-28 21:10:42 +0000 |
commit | ac87bfc370ec15c9c81f8738659fb2582b14b792 (patch) | |
tree | 2d2cd21cfc4faf885b5bf19b9c2f1f78add5c013 /gst/modplug/libmodplug/load_669.cpp | |
parent | 3b68b42af8feb78fe37cfadb6624e483f2e11d7d (diff) | |
download | gst-plugins-bad-ac87bfc370ec15c9c81f8738659fb2582b14b792.tar.gz gst-plugins-bad-ac87bfc370ec15c9c81f8738659fb2582b14b792.tar.bz2 gst-plugins-bad-ac87bfc370ec15c9c81f8738659fb2582b14b792.zip |
adding modplug
Original commit message from CVS:
adding modplug
Diffstat (limited to 'gst/modplug/libmodplug/load_669.cpp')
-rw-r--r-- | gst/modplug/libmodplug/load_669.cpp | 186 |
1 files changed, 186 insertions, 0 deletions
diff --git a/gst/modplug/libmodplug/load_669.cpp b/gst/modplug/libmodplug/load_669.cpp new file mode 100644 index 00000000..a0dbfabd --- /dev/null +++ b/gst/modplug/libmodplug/load_669.cpp @@ -0,0 +1,186 @@ +/* + * This source code is public domain. + * + * Authors: Olivier Lapicque <olivierl@jps.net>, + * Adam Goode <adam@evdebs.org> (endian and char fixes for PPC) +*/ + +//////////////////////////////////////////////////////////// +// 669 Composer / UNIS 669 module loader +//////////////////////////////////////////////////////////// + +#include "stdafx.h" +#include "sndfile.h" + +//#pragma warning(disable:4244) + +typedef struct tagFILEHEADER669 +{ + WORD sig; // 'if' or 'JN' + signed char songmessage[108]; // Song Message + BYTE samples; // number of samples (1-64) + BYTE patterns; // number of patterns (1-128) + BYTE restartpos; + BYTE orders[128]; + BYTE tempolist[128]; + BYTE breaks[128]; +} FILEHEADER669; + + +typedef struct tagSAMPLE669 +{ + BYTE filename[13]; + BYTE length[4]; // when will somebody think about DWORD align ??? + BYTE loopstart[4]; + BYTE loopend[4]; +} SAMPLE669; + + +BOOL CSoundFile::Read669(const BYTE *lpStream, DWORD dwMemLength) +//--------------------------------------------------------------- +{ + BOOL b669Ext; + const FILEHEADER669 *pfh = (const FILEHEADER669 *)lpStream; + const SAMPLE669 *psmp = (const SAMPLE669 *)(lpStream + 0x1F1); + DWORD dwMemPos = 0; + + if ((!lpStream) || (dwMemLength < sizeof(FILEHEADER669))) return FALSE; + if ((bswapLE16(pfh->sig) != 0x6669) && (bswapLE16(pfh->sig) != 0x4E4A)) return FALSE; + b669Ext = (bswapLE16(pfh->sig) == 0x4E4A) ? TRUE : FALSE; + if ((!pfh->samples) || (pfh->samples > 64) || (pfh->restartpos >= 128) + || (!pfh->patterns) || (pfh->patterns > 128)) return FALSE; + DWORD dontfuckwithme = 0x1F1 + pfh->samples * sizeof(SAMPLE669) + pfh->patterns * 0x600; + if (dontfuckwithme > dwMemLength) return FALSE; + for (UINT ichk=0; ichk<pfh->samples; ichk++) + { + DWORD len = bswapLE32(*((DWORD *)(&psmp[ichk].length))); + dontfuckwithme += len; + } + if (dontfuckwithme > dwMemLength) return FALSE; + // That should be enough checking: this must be a 669 module. + m_nType = MOD_TYPE_669; + m_dwSongFlags |= SONG_LINEARSLIDES; + m_nMinPeriod = 28 << 2; + m_nMaxPeriod = 1712 << 3; + m_nDefaultTempo = 125; + m_nDefaultSpeed = 6; + m_nChannels = 8; + memcpy(m_szNames[0], pfh->songmessage, 16); + m_nSamples = pfh->samples; + for (UINT nins=1; nins<=m_nSamples; nins++, psmp++) + { + DWORD len = bswapLE32(*((DWORD *)(&psmp->length))); + DWORD loopstart = bswapLE32(*((DWORD *)(&psmp->loopstart))); + DWORD loopend = bswapLE32(*((DWORD *)(&psmp->loopend))); + if (len > MAX_SAMPLE_LENGTH) len = MAX_SAMPLE_LENGTH; + if ((loopend > len) && (!loopstart)) loopend = 0; + if (loopend > len) loopend = len; + if (loopstart + 4 >= loopend) loopstart = loopend = 0; + Ins[nins].nLength = len; + Ins[nins].nLoopStart = loopstart; + Ins[nins].nLoopEnd = loopend; + if (loopend) Ins[nins].uFlags |= CHN_LOOP; + memcpy(m_szNames[nins], psmp->filename, 13); + Ins[nins].nVolume = 256; + Ins[nins].nGlobalVol = 64; + Ins[nins].nPan = 128; + } + // Song Message + m_lpszSongComments = new char[109]; + memcpy(m_lpszSongComments, pfh->songmessage, 108); + m_lpszSongComments[108] = 0; + // Reading Orders + memcpy(Order, pfh->orders, 128); + m_nRestartPos = pfh->restartpos; + if (Order[m_nRestartPos] >= pfh->patterns) m_nRestartPos = 0; + // Reading Pattern Break Locations + for (UINT npan=0; npan<8; npan++) + { + ChnSettings[npan].nPan = (npan & 1) ? 0x30 : 0xD0; + ChnSettings[npan].nVolume = 64; + } + // Reading Patterns + dwMemPos = 0x1F1 + pfh->samples * 25; + for (UINT npat=0; npat<pfh->patterns; npat++) + { + Patterns[npat] = AllocatePattern(64, m_nChannels); + if (!Patterns[npat]) break; + PatternSize[npat] = 64; + MODCOMMAND *m = Patterns[npat]; + const BYTE *p = lpStream + dwMemPos; + for (UINT row=0; row<64; row++) + { + MODCOMMAND *mspeed = m; + if ((row == pfh->breaks[npat]) && (row != 63)) + { + for (UINT i=0; i<8; i++) + { + m[i].command = CMD_PATTERNBREAK; + m[i].param = 0; + } + } + for (UINT n=0; n<8; n++, m++, p+=3) + { + UINT note = p[0] >> 2; + UINT instr = ((p[0] & 0x03) << 4) | (p[1] >> 4); + UINT vol = p[1] & 0x0F; + if (p[0] < 0xFE) + { + m->note = note + 37; + m->instr = instr + 1; + } + if (p[0] <= 0xFE) + { + m->volcmd = VOLCMD_VOLUME; + m->vol = (vol << 2) + 2; + } + if (p[2] != 0xFF) + { + UINT command = p[2] >> 4; + UINT param = p[2] & 0x0F; + switch(command) + { + case 0x00: command = CMD_PORTAMENTOUP; break; + case 0x01: command = CMD_PORTAMENTODOWN; break; + case 0x02: command = CMD_TONEPORTAMENTO; break; + case 0x03: command = CMD_MODCMDEX; param |= 0x50; break; + case 0x04: command = CMD_VIBRATO; param |= 0x40; break; + case 0x05: if (param) command = CMD_SPEED; else command = 0; param += 2; break; + case 0x06: if (param == 0) { command = CMD_PANNINGSLIDE; param = 0xFE; } else + if (param == 1) { command = CMD_PANNINGSLIDE; param = 0xEF; } else + command = 0; + break; + default: command = 0; + } + if (command) + { + if (command == CMD_SPEED) mspeed = NULL; + m->command = command; + m->param = param; + } + } + } + if ((!row) && (mspeed)) + { + for (UINT i=0; i<8; i++) if (!mspeed[i].command) + { + mspeed[i].command = CMD_SPEED; + mspeed[i].param = pfh->tempolist[npat] + 2; + break; + } + } + } + dwMemPos += 0x600; + } + // Reading Samples + for (UINT n=1; n<=m_nSamples; n++) + { + UINT len = Ins[n].nLength; + if (dwMemPos >= dwMemLength) break; + if (len > 4) ReadSample(&Ins[n], RS_PCM8U, (LPSTR)(lpStream+dwMemPos), dwMemLength - dwMemPos); + dwMemPos += len; + } + return TRUE; +} + + |