Mercurial > audlegacy-plugins
view src/modplug/load_mod.cxx @ 3189:ab6c7ebcd301
alsa-ng: Only support 16bit output for now. Someone else can debug this crap.
author | William Pitcock <nenolod@atheme.org> |
---|---|
date | Fri, 19 Jun 2009 09:14:22 -0500 |
parents | 107c1fed3d92 |
children |
line wrap: on
line source
/* * This source code is public domain. * * Authors: Olivier Lapicque <olivierl@jps.net>, * Adam Goode <adam@evdebs.org> (endian and char fixes for PPC) */ #include "stdafx.h" #include "sndfile.h" //#pragma warning(disable:4244) extern WORD ProTrackerPeriodTable[6*12]; ////////////////////////////////////////////////////////// // ProTracker / NoiseTracker MOD/NST file support void CSoundFile::ConvertModCommand(MODCOMMAND *m) const //----------------------------------------------------- { UINT command = m->command, param = m->param; switch(command) { case 0x00: if (param) command = CMD_ARPEGGIO; break; case 0x01: command = CMD_PORTAMENTOUP; break; case 0x02: command = CMD_PORTAMENTODOWN; break; case 0x03: command = CMD_TONEPORTAMENTO; break; case 0x04: command = CMD_VIBRATO; break; case 0x05: command = CMD_TONEPORTAVOL; if (param & 0xF0) param &= 0xF0; break; case 0x06: command = CMD_VIBRATOVOL; if (param & 0xF0) param &= 0xF0; break; case 0x07: command = CMD_TREMOLO; break; case 0x08: command = CMD_PANNING8; break; case 0x09: command = CMD_OFFSET; break; case 0x0A: command = CMD_VOLUMESLIDE; if (param & 0xF0) param &= 0xF0; break; case 0x0B: command = CMD_POSITIONJUMP; break; case 0x0C: command = CMD_VOLUME; break; case 0x0D: command = CMD_PATTERNBREAK; param = ((param >> 4) * 10) + (param & 0x0F); break; case 0x0E: command = CMD_MODCMDEX; break; case 0x0F: command = (param <= (UINT)((m_nType & (MOD_TYPE_XM|MOD_TYPE_MT2)) ? 0x1F : 0x20)) ? CMD_SPEED : CMD_TEMPO; if ((param == 0xFF) && (m_nSamples == 15)) command = 0; break; // Extension for XM extended effects case 'G' - 55: command = CMD_GLOBALVOLUME; break; case 'H' - 55: command = CMD_GLOBALVOLSLIDE; if (param & 0xF0) param &= 0xF0; break; case 'K' - 55: command = CMD_KEYOFF; break; case 'L' - 55: command = CMD_SETENVPOSITION; break; case 'M' - 55: command = CMD_CHANNELVOLUME; break; case 'N' - 55: command = CMD_CHANNELVOLSLIDE; break; case 'P' - 55: command = CMD_PANNINGSLIDE; if (param & 0xF0) param &= 0xF0; break; case 'R' - 55: command = CMD_RETRIG; break; case 'T' - 55: command = CMD_TREMOR; break; case 'X' - 55: command = CMD_XFINEPORTAUPDOWN; break; case 'Y' - 55: command = CMD_PANBRELLO; break; case 'Z' - 55: command = CMD_MIDI; break; default: command = 0; } m->command = command; m->param = param; } WORD CSoundFile::ModSaveCommand(const MODCOMMAND *m, BOOL bXM) const //------------------------------------------------------------------ { UINT command = m->command & 0x3F, param = m->param; switch(command) { case 0: command = param = 0; break; case CMD_ARPEGGIO: command = 0; break; case CMD_PORTAMENTOUP: if (m_nType & (MOD_TYPE_S3M|MOD_TYPE_IT|MOD_TYPE_STM)) { if ((param & 0xF0) == 0xE0) { command=0x0E; param=((param & 0x0F) >> 2)|0x10; break; } else if ((param & 0xF0) == 0xF0) { command=0x0E; param &= 0x0F; param|=0x10; break; } } command = 0x01; break; case CMD_PORTAMENTODOWN: if (m_nType & (MOD_TYPE_S3M|MOD_TYPE_IT|MOD_TYPE_STM)) { if ((param & 0xF0) == 0xE0) { command=0x0E; param=((param & 0x0F) >> 2)|0x20; break; } else if ((param & 0xF0) == 0xF0) { command=0x0E; param &= 0x0F; param|=0x20; break; } } command = 0x02; break; case CMD_TONEPORTAMENTO: command = 0x03; break; case CMD_VIBRATO: command = 0x04; break; case CMD_TONEPORTAVOL: command = 0x05; break; case CMD_VIBRATOVOL: command = 0x06; break; case CMD_TREMOLO: command = 0x07; break; case CMD_PANNING8: command = 0x08; if (bXM) { if ((m_nType != MOD_TYPE_IT) && (m_nType != MOD_TYPE_XM) && (param <= 0x80)) { param <<= 1; if (param > 255) param = 255; } } else { if ((m_nType == MOD_TYPE_IT) || (m_nType == MOD_TYPE_XM)) param >>= 1; } break; case CMD_OFFSET: command = 0x09; break; case CMD_VOLUMESLIDE: command = 0x0A; break; case CMD_POSITIONJUMP: command = 0x0B; break; case CMD_VOLUME: command = 0x0C; break; case CMD_PATTERNBREAK: command = 0x0D; param = ((param / 10) << 4) | (param % 10); break; case CMD_MODCMDEX: command = 0x0E; break; case CMD_SPEED: command = 0x0F; if (param > 0x20) param = 0x20; break; case CMD_TEMPO: if (param > 0x20) { command = 0x0F; break; } return 0; case CMD_GLOBALVOLUME: command = 'G' - 55; break; case CMD_GLOBALVOLSLIDE: command = 'H' - 55; break; case CMD_KEYOFF: command = 'K' - 55; break; case CMD_SETENVPOSITION: command = 'L' - 55; break; case CMD_CHANNELVOLUME: command = 'M' - 55; break; case CMD_CHANNELVOLSLIDE: command = 'N' - 55; break; case CMD_PANNINGSLIDE: command = 'P' - 55; break; case CMD_RETRIG: command = 'R' - 55; break; case CMD_TREMOR: command = 'T' - 55; break; case CMD_XFINEPORTAUPDOWN: command = 'X' - 55; break; case CMD_PANBRELLO: command = 'Y' - 55; break; case CMD_MIDI: command = 'Z' - 55; break; case CMD_S3MCMDEX: switch(param & 0xF0) { case 0x10: command = 0x0E; param = (param & 0x0F) | 0x30; break; case 0x20: command = 0x0E; param = (param & 0x0F) | 0x50; break; case 0x30: command = 0x0E; param = (param & 0x0F) | 0x40; break; case 0x40: command = 0x0E; param = (param & 0x0F) | 0x70; break; case 0x90: command = 'X' - 55; break; case 0xB0: command = 0x0E; param = (param & 0x0F) | 0x60; break; case 0xA0: case 0x50: case 0x70: case 0x60: command = param = 0; break; default: command = 0x0E; break; } break; default: command = param = 0; } return (WORD)((command << 8) | (param)); } #pragma pack(1) typedef struct _MODSAMPLE { CHAR name[22]; WORD length; BYTE finetune; BYTE volume; WORD loopstart; WORD looplen; } MODSAMPLE, *PMODSAMPLE; typedef struct _MODMAGIC { BYTE nOrders; BYTE nRestartPos; BYTE Orders[128]; char Magic[4]; // changed from CHAR } MODMAGIC, *PMODMAGIC; #pragma pack() BOOL IsMagic(LPCSTR s1, LPCSTR s2) { return ((*(DWORD *)s1) == (*(DWORD *)s2)) ? TRUE : FALSE; } BOOL CSoundFile::ReadMod(const BYTE *lpStream, DWORD dwMemLength) //--------------------------------------------------------------- { char s[1024]; // changed from CHAR DWORD dwMemPos, dwTotalSampleLen; PMODMAGIC pMagic; UINT nErr; if ((!lpStream) || (dwMemLength < 0x600)) return FALSE; dwMemPos = 20; m_nSamples = 31; m_nChannels = 4; pMagic = (PMODMAGIC)(lpStream+dwMemPos+sizeof(MODSAMPLE)*31); // Check Mod Magic memcpy(s, pMagic->Magic, 4); if ((IsMagic(s, "M.K.")) || (IsMagic(s, "M!K!")) || (IsMagic(s, "M&K!")) || (IsMagic(s, "N.T."))) m_nChannels = 4; else if ((IsMagic(s, "CD81")) || (IsMagic(s, "OKTA"))) m_nChannels = 8; else if ((s[0]=='F') && (s[1]=='L') && (s[2]=='T') && (s[3]>='4') && (s[3]<='9')) m_nChannels = s[3] - '0'; else if ((s[0]>='4') && (s[0]<='9') && (s[1]=='C') && (s[2]=='H') && (s[3]=='N')) m_nChannels = s[0] - '0'; else if ((s[0]=='1') && (s[1]>='0') && (s[1]<='9') && (s[2]=='C') && (s[3]=='H')) m_nChannels = s[1] - '0' + 10; else if ((s[0]=='2') && (s[1]>='0') && (s[1]<='9') && (s[2]=='C') && (s[3]=='H')) m_nChannels = s[1] - '0' + 20; else if ((s[0]=='3') && (s[1]>='0') && (s[1]<='2') && (s[2]=='C') && (s[3]=='H')) m_nChannels = s[1] - '0' + 30; else if ((s[0]=='T') && (s[1]=='D') && (s[2]=='Z') && (s[3]>='4') && (s[3]<='9')) m_nChannels = s[3] - '0'; else if (IsMagic(s,"16CN")) m_nChannels = 16; else if (IsMagic(s,"32CN")) m_nChannels = 32; else m_nSamples = 15; // Load Samples nErr = 0; dwTotalSampleLen = 0; for (UINT i=1; i<=m_nSamples; i++) { PMODSAMPLE pms = (PMODSAMPLE)(lpStream+dwMemPos); MODINSTRUMENT *psmp = &Ins[i]; UINT loopstart, looplen; memcpy(m_szNames[i], pms->name, 22); m_szNames[i][22] = 0; psmp->uFlags = 0; psmp->nLength = bswapBE16(pms->length)*2; dwTotalSampleLen += psmp->nLength; psmp->nFineTune = MOD2XMFineTune(pms->finetune & 0x0F); psmp->nVolume = 4*pms->volume; if (psmp->nVolume > 256) { psmp->nVolume = 256; nErr++; } psmp->nGlobalVol = 64; psmp->nPan = 128; loopstart = bswapBE16(pms->loopstart)*2; looplen = bswapBE16(pms->looplen)*2; // Fix loops if ((looplen > 2) && (loopstart+looplen > psmp->nLength) && (loopstart/2+looplen <= psmp->nLength)) { loopstart /= 2; } psmp->nLoopStart = loopstart; psmp->nLoopEnd = loopstart + looplen; if (psmp->nLength < 2) psmp->nLength = 0; if (psmp->nLength) { UINT derr = 0; if (psmp->nLoopStart >= psmp->nLength) { psmp->nLoopStart = psmp->nLength-1; derr|=1; } if (psmp->nLoopEnd > psmp->nLength) { psmp->nLoopEnd = psmp->nLength; derr |= 1; } if (psmp->nLoopStart > psmp->nLoopEnd) derr |= 1; if (psmp->nLoopEnd > psmp->nLoopStart) { psmp->uFlags |= CHN_LOOP; } } dwMemPos += sizeof(MODSAMPLE); } if ((m_nSamples == 15) && (dwTotalSampleLen > dwMemLength * 4)) return FALSE; pMagic = (PMODMAGIC)(lpStream+dwMemPos); dwMemPos += sizeof(MODMAGIC); if (m_nSamples == 15) dwMemPos -= 4; memset(Order, 0,sizeof(Order)); memcpy(Order, pMagic->Orders, 128); UINT nbp, nbpbuggy, nbpbuggy2, norders; norders = pMagic->nOrders; if ((!norders) || (norders > 0x80)) { norders = 0x80; while ((norders > 1) && (!Order[norders-1])) norders--; } nbpbuggy = 0; nbpbuggy2 = 0; nbp = 0; for (UINT iord=0; iord<128; iord++) { UINT i = Order[iord]; if ((i < 0x80) && (nbp <= i)) { nbp = i+1; if (iord<norders) nbpbuggy = nbp; } if (i >= nbpbuggy2) nbpbuggy2 = i+1; } for (UINT iend=norders; iend<MAX_ORDERS; iend++) Order[iend] = 0xFF; norders--; m_nRestartPos = pMagic->nRestartPos; if (m_nRestartPos >= 0x78) m_nRestartPos = 0; if (m_nRestartPos + 1 >= (UINT)norders) m_nRestartPos = 0; if (!nbp) return FALSE; DWORD dwWowTest = dwTotalSampleLen+dwMemPos; if ((IsMagic(pMagic->Magic, "M.K.")) && (dwWowTest + nbp*8*256 == dwMemLength)) m_nChannels = 8; if ((nbp != nbpbuggy) && (dwWowTest + nbp*m_nChannels*256 != dwMemLength)) { if (dwWowTest + nbpbuggy*m_nChannels*256 == dwMemLength) nbp = nbpbuggy; else nErr += 8; } else if ((nbpbuggy2 > nbp) && (dwWowTest + nbpbuggy2*m_nChannels*256 == dwMemLength)) { nbp = nbpbuggy2; } if ((dwWowTest < 0x600) || (dwWowTest > dwMemLength)) nErr += 8; if ((m_nSamples == 15) && (nErr >= 16)) return FALSE; // Default settings m_nType = MOD_TYPE_MOD; m_nDefaultSpeed = 6; m_nDefaultTempo = 125; m_nMinPeriod = 14 << 2; m_nMaxPeriod = 3424 << 2; memcpy(m_szNames, lpStream, 20); // Setting channels pan for (UINT ich=0; ich<m_nChannels; ich++) { ChnSettings[ich].nVolume = 64; ChnSettings[ich].nPan = (((ich&3)==1) || ((ich&3)==2)) ? 256 : 0; } m_nStereoSeparation = (gdwSoundSetup & SNDMIX_MAXDEFAULTPAN) ? 128 : 64; // Reading channels for (UINT ipat=0; ipat<nbp; ipat++) { if (ipat < MAX_PATTERNS) { if ((Patterns[ipat] = AllocatePattern(64, m_nChannels)) == NULL) break; PatternSize[ipat] = 64; PatternAllocSize[ipat] = 64; if (dwMemPos + m_nChannels*256 >= dwMemLength) break; MODCOMMAND *m = Patterns[ipat]; LPCBYTE p = lpStream + dwMemPos; for (UINT j=m_nChannels*64; j; m++,p+=4,j--) { BYTE A0=p[0], A1=p[1], A2=p[2], A3=p[3]; UINT n = ((((UINT)A0 & 0x0F) << 8) | (A1)); if ((n) && (n != 0xFFF)) m->note = GetNoteFromPeriod(n << 2); m->instr = ((UINT)A2 >> 4) | (A0 & 0x10); m->command = A2 & 0x0F; m->param = A3; if ((m->command) || (m->param)) ConvertModCommand(m); } } dwMemPos += m_nChannels*256; } // Reading instruments DWORD dwErrCheck = 0; for (UINT ismp=1; ismp<=m_nSamples; ismp++) if (Ins[ismp].nLength) { LPSTR p = (LPSTR)(lpStream+dwMemPos); UINT flags = 0; if (dwMemPos + 5 >= dwMemLength) break; if (!strnicmp(p, "ADPCM", 5)) { flags = 3; p += 5; dwMemPos += 5; } DWORD dwSize = ReadSample(&Ins[ismp], flags, p, dwMemLength - dwMemPos); if (dwSize) { dwMemPos += dwSize; dwErrCheck++; } } #ifdef MODPLUG_TRACKER return TRUE; #else return (dwErrCheck) ? TRUE : FALSE; #endif }