/** * Furnace Tracker - multi-system chiptune tracker * Copyright (C) 2021-2022 tildearrow and contributors * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #ifndef _SONG_H #define _SONG_H #include #include #define DIV_MAX_CHANS 128 #include "../ta-utils.h" #include "orders.h" #include "instrument.h" #include "pattern.h" #include "wavetable.h" #include "sample.h" enum DivSystem { DIV_SYSTEM_NULL=0, DIV_SYSTEM_YMU759, DIV_SYSTEM_GENESIS, // ** COMPOUND SYSTEM - DO NOT USE! ** DIV_SYSTEM_GENESIS_EXT, // ** COMPOUND SYSTEM - DO NOT USE! ** DIV_SYSTEM_SMS, DIV_SYSTEM_SMS_OPLL, // ** COMPOUND SYSTEM - DO NOT USE! ** DIV_SYSTEM_GB, DIV_SYSTEM_PCE, DIV_SYSTEM_NES, DIV_SYSTEM_NES_VRC7, // ** COMPOUND SYSTEM - DO NOT USE! ** DIV_SYSTEM_NES_FDS, // ** COMPOUND SYSTEM - DO NOT USE! ** DIV_SYSTEM_C64_6581, DIV_SYSTEM_C64_8580, DIV_SYSTEM_ARCADE, // ** COMPOUND SYSTEM - DO NOT USE! ** DIV_SYSTEM_YM2610, DIV_SYSTEM_YM2610_EXT, DIV_SYSTEM_AY8910, DIV_SYSTEM_AMIGA, DIV_SYSTEM_YM2151, DIV_SYSTEM_YM2612, DIV_SYSTEM_TIA, DIV_SYSTEM_SAA1099, DIV_SYSTEM_AY8930, DIV_SYSTEM_VIC20, DIV_SYSTEM_PET, DIV_SYSTEM_SNES, DIV_SYSTEM_VRC6, DIV_SYSTEM_OPLL, DIV_SYSTEM_FDS, DIV_SYSTEM_MMC5, DIV_SYSTEM_N163, DIV_SYSTEM_OPN, DIV_SYSTEM_OPN_EXT, DIV_SYSTEM_PC98, DIV_SYSTEM_PC98_EXT, DIV_SYSTEM_OPL, DIV_SYSTEM_OPL2, DIV_SYSTEM_OPL3, DIV_SYSTEM_MULTIPCM, DIV_SYSTEM_PCSPKR, DIV_SYSTEM_POKEY, DIV_SYSTEM_RF5C68, DIV_SYSTEM_SWAN, DIV_SYSTEM_OPZ, DIV_SYSTEM_POKEMINI, DIV_SYSTEM_SEGAPCM, DIV_SYSTEM_VBOY, DIV_SYSTEM_VRC7, DIV_SYSTEM_YM2610B, DIV_SYSTEM_SFX_BEEPER, DIV_SYSTEM_YM2612_EXT, DIV_SYSTEM_SCC, DIV_SYSTEM_OPL_DRUMS, DIV_SYSTEM_OPL2_DRUMS, DIV_SYSTEM_OPL3_DRUMS, DIV_SYSTEM_YM2610_FULL, DIV_SYSTEM_YM2610_FULL_EXT, DIV_SYSTEM_OPLL_DRUMS, DIV_SYSTEM_LYNX, DIV_SYSTEM_QSOUND, DIV_SYSTEM_VERA, DIV_SYSTEM_YM2610B_EXT, DIV_SYSTEM_SEGAPCM_COMPAT, DIV_SYSTEM_X1_010, DIV_SYSTEM_BUBSYS_WSG, DIV_SYSTEM_OPL4, DIV_SYSTEM_OPL4_DRUMS, DIV_SYSTEM_ES5506, DIV_SYSTEM_Y8950, DIV_SYSTEM_Y8950_DRUMS, DIV_SYSTEM_SCC_PLUS, DIV_SYSTEM_SOUND_UNIT, DIV_SYSTEM_MSM6295, DIV_SYSTEM_MSM6258, DIV_SYSTEM_YMZ280B, DIV_SYSTEM_NAMCO, DIV_SYSTEM_NAMCO_15XX, DIV_SYSTEM_NAMCO_CUS30, DIV_SYSTEM_YM2612_FRAC, DIV_SYSTEM_YM2612_FRAC_EXT, DIV_SYSTEM_RESERVED_8, DIV_SYSTEM_T6W28, DIV_SYSTEM_PCM_DAC, DIV_SYSTEM_DUMMY }; struct DivSubSong { String name, notes; unsigned char hilightA, hilightB; unsigned char timeBase, speed1, speed2, arpLen; short virtualTempoN, virtualTempoD; bool pal; bool customTempo; float hz; int patLen, ordersLen; DivOrders orders; DivChannelData pat[DIV_MAX_CHANS]; bool chanShow[DIV_MAX_CHANS]; unsigned char chanCollapse[DIV_MAX_CHANS]; String chanName[DIV_MAX_CHANS]; String chanShortName[DIV_MAX_CHANS]; void clearData(); void optimizePatterns(); void rearrangePatterns(); DivSubSong(): hilightA(4), hilightB(16), timeBase(0), speed1(6), speed2(6), arpLen(1), virtualTempoN(150), virtualTempoD(150), pal(true), customTempo(false), hz(60.0), patLen(64), ordersLen(1) { for (int i=0; i game name // writer -> ripper // createdDate -> year String carrier, composer, vendor, category, writer, arranger, copyright, manGroup, manInfo, createdDate, revisionDate; // more VGM specific stuff String nameJ, authorJ, categoryJ, systemNameJ; // other things String notes; // module details int insLen, waveLen, sampleLen; float masterVol; float tuning; // compatibility flags bool limitSlides; // linear pitch // 0: not linear // 1: only pitch changes (04xy/E5xx) linear // 2: full linear unsigned char linearPitch; unsigned char pitchSlideSpeed; // loop behavior // 0: reset on loop // 1: fake reset on loop // 2: don't do anything on loop unsigned char loopModality; // cut/delay effect behavior // 0: strict (don't allow value higher than or equal to speed) // 1: broken (don't allow value higher than speed) // 2: lax (allow value higher than speed) unsigned char delayBehavior; // 0B/0D treatment // 0: normal (0B/0D accepted) // 1: old Furnace (first one accepted) // 2: DefleMask (0D takes priority over 0B) unsigned char jumpTreatment; bool properNoiseLayout; bool waveDutyIsVol; bool resetMacroOnPorta; bool legacyVolumeSlides; bool compatibleArpeggio; bool noteOffResetsSlides; bool targetResetsSlides; bool arpNonPorta; bool algMacroBehavior; bool brokenShortcutSlides; bool ignoreDuplicateSlides; bool stopPortaOnNoteOff; bool continuousVibrato; bool brokenDACMode; bool oneTickCut; bool newInsTriggersInPorta; bool arp0Reset; bool brokenSpeedSel; bool noSlidesOnFirstTick; bool rowResetsArpPos; bool ignoreJumpAtEnd; bool buggyPortaAfterSlide; bool gbInsAffectsEnvelope; bool sharedExtStat; bool ignoreDACModeOutsideIntendedChannel; bool e1e2AlsoTakePriority; bool newSegaPCM; bool fbPortaPause; bool snDutyReset; bool pitchMacroIsLinear; bool oldOctaveBoundary; bool noOPN2Vol; bool newVolumeScaling; bool volMacroLinger; bool brokenOutVol; bool e1e2StopOnSameNote; bool brokenPortaArp; bool snNoLowPeriods; bool disableSampleMacro; bool autoSystem; std::vector ins; std::vector wave; std::vector sample; std::vector subsong; DivInstrument nullIns, nullInsOPLL, nullInsOPL, nullInsOPLDrums, nullInsQSound; DivWavetable nullWave; DivSample nullSample; /** * clear orders and patterns. */ void clearSongData(); /** * clear instruments. */ void clearInstruments(); /** * clear wavetables. */ void clearWavetables(); /** * clear samples. */ void clearSamples(); /** * unloads the song, freeing all memory associated with it. * use before destroying the object. */ void unload(); DivSong(): version(0), isDMF(false), systemLen(2), name(""), author(""), systemName(""), carrier(""), composer(""), vendor(""), category(""), writer(""), arranger(""), copyright(""), manGroup(""), manInfo(""), createdDate(""), revisionDate(""), insLen(0), waveLen(0), sampleLen(0), masterVol(1.0f), tuning(440.0f), limitSlides(false), linearPitch(2), pitchSlideSpeed(4), loopModality(2), delayBehavior(2), jumpTreatment(0), properNoiseLayout(true), waveDutyIsVol(false), resetMacroOnPorta(false), legacyVolumeSlides(false), compatibleArpeggio(false), noteOffResetsSlides(true), targetResetsSlides(true), arpNonPorta(false), algMacroBehavior(false), brokenShortcutSlides(false), ignoreDuplicateSlides(false), stopPortaOnNoteOff(false), continuousVibrato(false), brokenDACMode(false), oneTickCut(false), newInsTriggersInPorta(true), arp0Reset(true), brokenSpeedSel(false), noSlidesOnFirstTick(false), rowResetsArpPos(false), ignoreJumpAtEnd(false), buggyPortaAfterSlide(false), gbInsAffectsEnvelope(true), sharedExtStat(true), ignoreDACModeOutsideIntendedChannel(false), e1e2AlsoTakePriority(false), newSegaPCM(true), fbPortaPause(false), snDutyReset(false), pitchMacroIsLinear(true), oldOctaveBoundary(false), noOPN2Vol(false), newVolumeScaling(true), volMacroLinger(true), brokenOutVol(false), e1e2StopOnSameNote(false), brokenPortaArp(false), snNoLowPeriods(false), disableSampleMacro(false), autoSystem(true) { for (int i=0; i<32; i++) { system[i]=DIV_SYSTEM_NULL; systemVol[i]=64; systemPan[i]=0; systemFlags[i]=0; } subsong.push_back(new DivSubSong); system[0]=DIV_SYSTEM_YM2612; system[1]=DIV_SYSTEM_SMS; // OPLL default instrument contest winner - piano_guitar_idk by Weeppiko nullInsOPLL.type=DIV_INS_OPLL; nullInsOPLL.fm.opllPreset=0; nullInsOPLL.fm.alg=0; nullInsOPLL.fm.fb=7; nullInsOPLL.fm.fms=1; nullInsOPLL.fm.ams=0; nullInsOPLL.fm.op[0].ar=15; nullInsOPLL.fm.op[0].dr=5; nullInsOPLL.fm.op[0].sl=3; nullInsOPLL.fm.op[0].rr=3; nullInsOPLL.fm.op[0].tl=40; nullInsOPLL.fm.op[0].ksl=0; nullInsOPLL.fm.op[0].mult=5; nullInsOPLL.fm.op[0].am=0; nullInsOPLL.fm.op[0].vib=1; nullInsOPLL.fm.op[0].ksr=0; nullInsOPLL.fm.op[0].ssgEnv=8; nullInsOPLL.fm.op[1].ar=15; nullInsOPLL.fm.op[1].dr=1; nullInsOPLL.fm.op[1].sl=11; nullInsOPLL.fm.op[1].rr=6; nullInsOPLL.fm.op[1].tl=0; nullInsOPLL.fm.op[1].ksl=0; nullInsOPLL.fm.op[1].mult=1; nullInsOPLL.fm.op[1].am=0; nullInsOPLL.fm.op[1].vib=0; nullInsOPLL.fm.op[1].ksr=0; nullInsOPLL.fm.op[1].ssgEnv=8; nullInsOPLL.name="This is a bug! Report!"; nullInsOPL.type=DIV_INS_OPL; nullInsOPL.fm.alg=0; nullInsOPL.fm.fb=7; nullInsOPL.fm.op[0].dr=2; nullInsOPL.fm.op[0].rr=7; nullInsOPL.fm.op[0].tl=22; nullInsOPL.fm.op[0].ksl=1; nullInsOPL.fm.op[0].mult=3; nullInsOPL.fm.op[1].tl=0; nullInsOPL.fm.op[1].dr=3; nullInsOPL.fm.op[1].rr=12; nullInsOPL.fm.op[1].mult=1; nullInsOPL.name="This is a bug! Report!"; nullInsOPL.fm.kickFreq=(1<<10)|576; nullInsOPL.fm.snareHatFreq=(1<<10)|672; nullInsOPL.fm.tomTopFreq=896; nullInsOPLDrums=nullInsOPL; nullInsOPLDrums.type=DIV_INS_OPL_DRUMS; nullInsOPLDrums.fm.fixedDrums=true; for (int i=0; i<4; i++) { nullInsOPLDrums.fm.op[i].am=0; nullInsOPLDrums.fm.op[i].vib=0; nullInsOPLDrums.fm.op[i].ksr=0; nullInsOPLDrums.fm.op[i].sus=0; nullInsOPLDrums.fm.op[i].ws=0; nullInsOPLDrums.fm.op[i].ksl=0; nullInsOPLDrums.fm.op[i].tl=0; } // snare nullInsOPLDrums.fm.op[0].ar=13; nullInsOPLDrums.fm.op[0].dr=8; nullInsOPLDrums.fm.op[0].sl=4; nullInsOPLDrums.fm.op[0].rr=8; nullInsOPLDrums.fm.op[0].mult=1; // tom nullInsOPLDrums.fm.op[1].ar=15; nullInsOPLDrums.fm.op[1].dr=8; nullInsOPLDrums.fm.op[1].sl=5; nullInsOPLDrums.fm.op[1].rr=9; nullInsOPLDrums.fm.op[1].mult=5; // top nullInsOPLDrums.fm.op[2].ar=10; nullInsOPLDrums.fm.op[2].dr=10; nullInsOPLDrums.fm.op[2].sl=5; nullInsOPLDrums.fm.op[2].rr=5; nullInsOPLDrums.fm.op[2].mult=1; nullInsOPLDrums.fm.op[2].ksr=1; // hi-hat nullInsOPLDrums.fm.op[3].ar=12; nullInsOPLDrums.fm.op[3].dr=8; nullInsOPLDrums.fm.op[3].sl=10; nullInsOPLDrums.fm.op[3].rr=7; nullInsOPLDrums.fm.op[3].mult=2; nullInsQSound.std.panLMacro.mode=true; } }; #endif