/** * Furnace Tracker - multi-system chiptune tracker * Copyright (C) 2021-2024 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. */ #include #include #include #include #include "engine.h" #include "../fileutils.h" #include "../ta-log.h" struct TiunaNew { short pitch=-1; signed char ins=-1; signed char vol=-1; short sync=-1; }; struct TiunaLast { short pitch=0; signed char ins=0; signed char vol=0; int tick=1; bool forcePitch=true; }; struct TiunaCmd { signed char pitchChange=-1; short pitchSet=-1; signed char ins=-1; signed char vol=-1; short sync=-1; short wait=-1; }; struct TiunaBytes { unsigned char ch=0; int ticks=0; unsigned char size=0; unsigned char buf[16]; friend bool operator==(const TiunaBytes& l, const TiunaBytes& r) { if (l.size!=r.size) return false; if (l.ticks!=r.ticks) return false; return memcmp(l.buf,r.buf,l.size)==0; } }; struct TiunaMatch { int pos; int endPos; int size; int id; }; struct TiunaMatches { int bytesSaved=INT32_MIN; int length=0; int ticks=0; std::vector pos; }; static void writeCmd(std::vector& cmds, TiunaCmd& cmd, unsigned char ch, int& lastWait, int fromTick, int toTick) { while (fromTick=0) { nbuf.buf[nlen++]=0b00111100; nbuf.buf[nlen++]=cmd.sync; } if (cmd.wait>=17) { nbuf.buf[nlen++]=0b00111111; nbuf.buf[nlen++]=cmd.wait-1; } if (cmd.pitchChange>=0) { nbuf.buf[nlen++]=0b01000000|cmd.pitchChange; } if (cmd.pitchSet>=0) { nbuf.buf[nlen++]=0b01100000|(cmd.pitchSet>>8); nbuf.buf[nlen++]=cmd.pitchSet&0xff; } if (cmd.vol>=1) { vcw1|=0x40|cmd.vol; vcwLen++; } if (cmd.ins>=0) { vcw1|=0x20; if (vcwLen==0) vcw1|=cmd.ins; else vcw2=cmd.ins; vcwLen++; } if (cmd.wait>=1 && cmd.wait<17) { unsigned char val=cmd.wait-1; vcw1|=0x10; if (vcwLen==0) vcw1|=val; else if (vcwLen==1) vcw2=val; else vcw2|=(val<<4); vcwLen++; } nbuf.buf[nlen++]=vcw1; if (vcwLen>=2) nbuf.buf[nlen++]=vcw2; nbuf.size=nlen; cmds.push_back(nbuf); cmd=TiunaCmd(); fromTick+=val; } } SafeWriter* DivEngine::saveTiuna(const bool* sysToExport, const char* baseLabel, int firstBankSize, int otherBankSize) { stop(); repeatPattern=false; shallStop=false; setOrder(0); BUSY_BEGIN_SOFT; // determine loop point // bool stopped=false; int loopOrder=0; int loopOrderRow=0; int loopEnd=0; walkSong(loopOrder,loopOrderRow,loopEnd); logI("loop point: %d %d",loopOrder,loopOrderRow); SafeWriter* w=new SafeWriter; w->init(); int tiaIdx=-1; for (int i=0; itoggleRegisterDump(true); break; } } if (tiaIdx<0) { lastError="selected TIA system not found"; return NULL; } // write patterns // bool writeLoop=false; bool done=false; playSub(false); int tick=0; // int loopTick=-1; TiunaLast last[2]; TiunaNew news[2]; std::map allCmds[2]; while (!done) { // TODO implement loop // if (loopTick<0 && loopOrder==curOrder && loopOrderRow==curRow // && (ticks-((tempoAccum+virtualTempoN)/virtualTempoD))<=0 // ) { // writeLoop=true; // loopTick=tick; // // invalidate last register state so it always force an absolute write after loop // for (int i=0; i<2; i++) { // last[i]=TiunaLast(); // last[i].pitch=-1; // last[i].ins=-1; // last[i].vol=-1; // } // } if (nextTick(false,true) || !playing) { // stopped=!playing; done=true; break; } for (int i=0; i<2; i++) { news[i]=TiunaNew(); } // get register dumps std::vector& writes=disCont[tiaIdx].dispatch->getRegisterWrites(); for (const DivRegWrite& i: writes) { switch (i.addr) { case 0xfffe0000: case 0xfffe0001: news[i.addr&1].pitch=i.val; break; case 0xfffe0002: news[0].sync=i.val; break; case 0x15: case 0x16: news[i.addr-0x15].ins=i.val; break; case 0x19: case 0x1a: news[i.addr-0x19].vol=i.val; break; default: break; } } writes.clear(); // collect changes for (int i=0; i<2; i++) { TiunaCmd cmds; bool hasCmd=false; if (news[i].pitch>=0 && (last[i].forcePitch || news[i].pitch!=last[i].pitch)) { int dt=news[i].pitch-last[i].pitch; if (!last[i].forcePitch && abs(dt)<=16) { if (dt<0) cmds.pitchChange=15-dt; else cmds.pitchChange=dt-1; } else cmds.pitchSet=news[i].pitch; last[i].pitch=news[i].pitch; last[i].forcePitch=false; hasCmd=true; } if (news[i].ins>=0 && news[i].ins!=last[i].ins) { cmds.ins=news[i].ins; last[i].ins=news[i].ins; hasCmd=true; } if (news[i].vol>=0 && news[i].vol!=last[i].vol) { cmds.vol=(news[i].vol-last[i].vol)&0xf; last[i].vol=news[i].vol; hasCmd=true; } if (news[i].sync>=0) { cmds.sync=news[i].sync; hasCmd=true; } if (hasCmd) allCmds[i][tick]=cmds; } cmdStream.clear(); tick++; } for (int i=0; igetRegisterWrites().clear(); disCont[i].dispatch->toggleRegisterDump(false); } remainingLoops=-1; playing=false; freelance=false; extValuePresent=false; BUSY_END; // render commands std::vector renderedCmds; w->writeText(fmt::format( "; Generated by Furnace " DIV_VERSION "\n" "; Name: {}\n" "; Author: {}\n" "; Album: {}\n" "; Subsong #{}: {}\n\n", song.name,song.author,song.category,curSubSongIndex+1,curSubSong->name )); for (int i=0; i<2; i++) { TiunaCmd lastCmd; int lastTick=0; int lastWait=0; // bool looped=false; for (auto& kv: allCmds[i]) { // if (!looped && !stopped && loopTick>=0 && kv.first>=loopTick) { // writeCmd(w,&lastCmd,&lastWait,loopTick-lastTick); // w->writeText(".loop\n"); // lastTick=loopTick; // looped=true; // } writeCmd(renderedCmds,lastCmd,i,lastWait,lastTick,kv.first); lastTick=kv.first; lastCmd=kv.second; } writeCmd(renderedCmds,lastCmd,i,lastWait,lastTick,tick); // if (stopped || loopTick<0) w->writeText(".loop\n db 0\n"); } // compress commands std::vector confirmedMatches; std::vector callTicks; int cmId=0; int cmdSize=renderedCmds.size(); std::vector processed=std::vector(cmdSize,false); while (firstBankSize>768 && cmId<(MAX(firstBankSize/1024,1))*256) { std::map potentialMatches; for (int i=0; i=cmdSize-1) break; std::vector match; int ch=renderedCmds[i].ch; for (int j=i+1; j=cmdSize) break; int k=0; int ticks=0; int size=0; while ( (i+k)2) match.push_back({j,j+k,size,0}); if (k==0) k++; j+=k; } if (match.empty()) { i++; continue; } // find a length that results in most bytes saved TiunaMatches matches; int curSize=0; int curLength=1; int curTicks=0; while (true) { int bytesSaved=-4; bool found=false; for (const TiunaMatch& j: match) { if ((j.endPos-j.pos)>=curLength) { if (!found) { found=true; curSize+=renderedCmds[i+curLength-1].size; curTicks+=renderedCmds[i+curLength-1].ticks; } bytesSaved+=curSize-2; } } if (!found) break; if (bytesSaved>matches.bytesSaved) { matches.length=curLength; matches.bytesSaved=bytesSaved; matches.ticks=curTicks; } curLength++; } if (matches.bytesSaved>0) { matches.pos.push_back(i); for (const TiunaMatch& j: match) { if ((j.endPos-j.pos)>=matches.length) { matches.pos.push_back(j.pos); } } potentialMatches[i]=matches; } i++; } if (potentialMatches.empty()) break; int maxPMIdx=0; int maxPMVal=0; for (const auto& i: potentialMatches) { if (i.second.bytesSaved>maxPMVal) { maxPMVal=i.second.bytesSaved; maxPMIdx=i.first; } } int maxPMLen=potentialMatches[maxPMIdx].length; for (const int i: potentialMatches[maxPMIdx].pos) { confirmedMatches.push_back({i,i+maxPMLen,0,cmId}); std::fill(processed.begin()+i,processed.begin()+(i+maxPMLen),true); } callTicks.push_back(potentialMatches[maxPMIdx].ticks); logI("CM %04x added: pos=%d,len=%d,matches=%d,saved=%d",cmId,maxPMIdx,maxPMLen,potentialMatches[maxPMIdx].pos.size(),maxPMVal); cmId++; } std::sort(confirmedMatches.begin(),confirmedMatches.end(),[](const TiunaMatch& l, const TiunaMatch& r){ return l.pos256 that don't fill up a page // as they tends to increase the final size due to page alignment int cmIdLen=cmId>256?(cmId&~255):cmId; // overlap check for (int i=1; i<(int)confirmedMatches.size(); i++) { if (confirmedMatches[i-1].endPos<=confirmedMatches[i].pos) continue; lastError="impossible overlap found in matches list, please report"; return NULL; } SafeWriter dbg; dbg.init(); dbg.writeText(fmt::format("renderedCmds size={}\n",renderedCmds.size())); for (const auto& i: confirmedMatches) { dbg.writeText(fmt::format("pos={},end={},id={}\n",i.pos,i.endPos,i.id,i.size)); } // write commands int totalSize=0; int cnt=cmIdLen; w->writeText(fmt::format(" .section {0}_bank0\n .align $100\n{0}_calltable",baseLabel)); while (cnt>0) { int cnt2=MIN(cnt,256); w->writeText("\n .byte "); for (int j=0; jwriteText(fmt::format("<{}_c{},",baseLabel,cmIdLen-cnt+j)); } for (int j=cnt2; j<256; j++) { w->writeText("0,"); } w->seek(-1,SEEK_CUR); w->writeText("\n .byte "); for (int j=0; jwriteText(fmt::format(">{}_c{},",baseLabel,cmIdLen-cnt+j)); } for (int j=cnt2; j<256; j++) { w->writeText("0,"); } w->seek(-1,SEEK_CUR); w->writeText("\n .byte "); for (int j=0; jwriteText(fmt::format("{}_c{}>>13,",baseLabel,cmIdLen-cnt+j)); } for (int j=cnt2; j<256; j++) { w->writeText("0,"); } w->seek(-1,SEEK_CUR); w->writeText("\n .byte "); for (int j=0; jwriteText(fmt::format("{},",callTicks[cmIdLen-cnt+j]&0xff)); } w->seek(-1,SEEK_CUR); totalSize+=768+cnt2; cnt-=cnt2; } w->writeC('\n'); if (totalSize>firstBankSize) { lastError="first bank is not large enough to contain call table"; return NULL; } int curBank=0; int bankSize=totalSize; int maxBankSize=firstBankSize; int curCh=-1; std::vector callVisited=std::vector(cmIdLen,false); auto cmIter=confirmedMatches.begin(); for (int i=0; i<(int)renderedCmds.size(); i++) { int writeCall=-1; TiunaBytes cmd=renderedCmds[i]; if (cmIter!=confirmedMatches.end() && i==cmIter->pos) { if (cmIter->idid]) { unsigned char idLo=cmIter->id&0xff; unsigned char idHi=cmIter->id>>8; cmd={cmd.ch,0,2,{idHi,idLo}}; i=cmIter->endPos-1; } else { writeCall=cmIter->id; callVisited[writeCall]=true; } } cmIter++; } if (cmd.ch!=curCh) { if (curCh>=0) { w->writeText(" .text x\"e0\"\n"); totalSize++; bankSize++; } if (bankSize+cmd.size>=maxBankSize) { maxBankSize=otherBankSize; curBank++; w->writeText(fmt::format(" .endsection\n\n .section {}_bank{}",baseLabel,curBank)); bankSize=0; } w->writeText(fmt::format("\n{}_ch{}\n",baseLabel,cmd.ch)); curCh=cmd.ch; } if (bankSize+cmd.size+1>=maxBankSize) { maxBankSize=otherBankSize; curBank++; w->writeText(fmt::format(" .text x\"c0\"\n .endsection\n\n .section {}_bank{}\n",baseLabel,curBank)); totalSize++; bankSize=0; } if (writeCall>=0) { w->writeText(fmt::format("{}_c{}\n",baseLabel,writeCall)); } w->writeText(" .text x\""); for (int j=0; jwriteText(fmt::format("{:02x}",cmd.buf[j])); } w->writeText("\"\n"); totalSize+=cmd.size; bankSize+=cmd.size; } w->writeText(" .text x\"e0\"\n .endsection\n"); totalSize++; logI("total size: %d bytes (%d banks)",totalSize,curBank+1); FILE* f=ps_fopen("confirmedMatches.txt","wb"); if (f!=NULL) { fwrite(dbg.getFinalBuf(),1,dbg.size(),f); fclose(f); } return w; }