/** * 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. */ #include "pet.h" #include "../engine.h" #include #define rWrite(a,v) {regPool[(a)]=(v)&0xff; if((a)==10) {chan.sreg=(v); chan.cnt=2;}} #define CHIP_DIVIDER 16 #define SAMP_DIVIDER 4 const char* regCheatSheet6522[]={ "T2L", "08", "SR", "0A", "ACR", "0B", NULL }; const char** DivPlatformPET::getRegisterSheet() { return regCheatSheet6522; } const char* DivPlatformPET::getEffectName(unsigned char effect) { switch (effect) { case 0x10: return "10xx: Change waveform"; break; } return NULL; } void DivPlatformPET::acquire(short* bufL, short* bufR, size_t start, size_t len) { // high-level emulation of 6522 shift register for now int t2=regPool[8]*2+4; if (((regPool[11]>>2)&7)==4) { for (size_t h=start; h0) { int adv=MIN(cycs,chan.cnt); chan.cnt-=adv; cycs-=adv; if (chan.cnt==0) { chan.out=(chan.sreg&1)*32767; chan.sreg=(chan.sreg>>1)|((chan.sreg&1)<<7); chan.cnt=t2; } } bufL[h]=chan.out; bufR[h]=chan.out; } } else { chan.out=0; for (size_t h=start; h0) { if (regPool[11]!=16) { rWrite(11,16); rWrite(10,chan.wave); } } else { rWrite(11,0); } } void DivPlatformPET::tick(bool sysTick) { chan.std.next(); if (chan.std.vol.had) { chan.outVol=chan.std.vol.val&chan.vol; writeOutVol(); } if (chan.std.arp.had) { if (!chan.inPorta) { if (chan.std.arp.mode) { chan.baseFreq=NOTE_PERIODIC(chan.std.arp.val); } else { chan.baseFreq=NOTE_PERIODIC(chan.note+chan.std.arp.val); } } chan.freqChanged=true; } else { if (chan.std.arp.mode && chan.std.arp.finished) { chan.baseFreq=NOTE_PERIODIC(chan.note); chan.freqChanged=true; } } if (chan.std.wave.had) { if (chan.wave!=chan.std.wave.val) { chan.wave=chan.std.wave.val; rWrite(10,chan.wave); } } if (chan.std.pitch.had) { chan.freqChanged=true; } if (chan.freqChanged || chan.keyOn || chan.keyOff) { chan.freq=parent->calcFreq(chan.baseFreq,chan.pitch,true,0,chan.std.pitch.val); if (chan.freq>257) chan.freq=257; if (chan.freq<2) chan.freq=2; rWrite(8,chan.freq-2); if (chan.keyOn) { if (!chan.std.vol.will) { chan.outVol=chan.vol; writeOutVol(); } chan.keyOn=false; } if (chan.keyOff) { rWrite(11,0); chan.keyOff=false; } chan.freqChanged=false; } } int DivPlatformPET::dispatch(DivCommand c) { switch (c.cmd) { case DIV_CMD_NOTE_ON: { DivInstrument* ins=parent->getIns(chan.ins,DIV_INS_PET); if (c.value!=DIV_NOTE_NULL) { chan.baseFreq=NOTE_PERIODIC(c.value); chan.freqChanged=true; chan.note=c.value; } chan.active=true; chan.keyOn=true; chan.std.init(ins); break; } case DIV_CMD_NOTE_OFF: chan.active=false; chan.keyOff=true; chan.std.init(NULL); break; case DIV_CMD_NOTE_OFF_ENV: case DIV_CMD_ENV_RELEASE: chan.std.release(); break; case DIV_CMD_INSTRUMENT: if (chan.ins!=c.value || c.value2==1) { chan.ins=c.value; } break; case DIV_CMD_VOLUME: if (chan.vol!=c.value) { chan.vol=c.value; if (!chan.std.vol.had) { chan.outVol=chan.vol; writeOutVol(); } } break; case DIV_CMD_GET_VOLUME: return chan.vol; break; case DIV_CMD_PITCH: chan.pitch=c.value; chan.freqChanged=true; break; case DIV_CMD_WAVE: chan.wave=c.value; rWrite(10,chan.wave); break; case DIV_CMD_NOTE_PORTA: { int destFreq=NOTE_PERIODIC(c.value2); bool return2=false; if (destFreq>chan.baseFreq) { chan.baseFreq+=c.value; if (chan.baseFreq>=destFreq) { chan.baseFreq=destFreq; return2=true; } } else { chan.baseFreq-=c.value; if (chan.baseFreq<=destFreq) { chan.baseFreq=destFreq; return2=true; } } chan.freqChanged=true; if (return2) { chan.inPorta=false; return 2; } break; } case DIV_CMD_LEGATO: chan.baseFreq=NOTE_PERIODIC(c.value+((chan.std.arp.will && !chan.std.arp.mode)?(chan.std.arp.val):(0))); chan.freqChanged=true; chan.note=c.value; break; case DIV_CMD_PRE_PORTA: if (chan.active && c.value2) { if (parent->song.resetMacroOnPorta) chan.std.init(parent->getIns(chan.ins,DIV_INS_PET)); } chan.inPorta=c.value; break; case DIV_CMD_GET_VOLMAX: return 1; break; case DIV_ALWAYS_SET_VOLUME: return 1; break; default: break; } return 1; } void DivPlatformPET::muteChannel(int ch, bool mute) { isMuted=mute; writeOutVol(); } void DivPlatformPET::forceIns() { chan.insChanged=true; chan.freqChanged=true; writeOutVol(); } void* DivPlatformPET::getChanState(int ch) { return &chan; } unsigned char* DivPlatformPET::getRegisterPool() { return regPool; } int DivPlatformPET::getRegisterPoolSize() { return 16; } void DivPlatformPET::reset() { memset(regPool,0,16); chan=Channel(); chan.std.setEngine(parent); } bool DivPlatformPET::isStereo() { return false; } void DivPlatformPET::notifyInsDeletion(void* ins) { chan.std.notifyInsDeletion((DivInstrument*)ins); } void DivPlatformPET::poke(unsigned int addr, unsigned short val) { rWrite(addr,val); } void DivPlatformPET::poke(std::vector& wlist) { for (DivRegWrite& i: wlist) rWrite(i.addr,i.val); } int DivPlatformPET::init(DivEngine* p, int channels, int sugRate, unsigned int flags) { parent=p; dumpWrites=false; skipRegisterWrites=false; chipClock=1000000; rate=chipClock/SAMP_DIVIDER; // = 250000kHz isMuted=false; reset(); return 1; } DivPlatformPET::~DivPlatformPET() { }