const IO = require('./common/io.js'); const BCD = require('./utils/bcd.js'); const BITM = require('./utils/bitman.js'); function EnableICBit(cia, pos){ v = 1 << pos; if ((cia.__ICMask & v) === 0){ cia.__IC = cia.__IC | v; // Enabling any bit enables bit 7 as well! cia.__IC = cia.__IC | 0x80; } } function TODTick(cia){ if (cia.__TODTick < 0){return;} cia.__TODTick += 1; if (cia.__TODTick === 6){ cia.__TODTick = (cia.__CTA & 0x80) >> 7; if (cia.__TOD[0] == 0x09){ cia.__TOD[0] = 0; if (cia.__TOD[1] == 0x59){ cia.__TOD[1] = 0; if (cia.__TOD[2] == 0x59){ cia.__TOD[2] = 0; if (cia.__TOD[3] == 0x11 || cia.__TOD[3] == 0x91){ cia.__TOD[3] = (cia.__TOD[3] == 0x91) ? 0x00 : 0x80; } else { let pm = cia.__TOD[3] & 0x80; cia.__TOD[3] = BCD.add(cia.__TOD[3] & 0x7F, 0x01, 2) | pm; } } else {cia.__TOD[2] = BCD.add(cia.__TOD[2], 0x01, 2);} } else {cia.__TOD[1] = BCD.add(cia.__TOD[1], 0x01, 2);} } else {cia.__TOD[0] = BCD.add(cia.__TOD[0], 0x01, 2);} } if (cia.__TOD[0] == cia.__ALTOD[0] && cia.__TOD[1] == cia.__ALTOD[1] && cia.__TOD[2] == cia.__ALTOD[2] && cia.__TOD[3] == cia.__ALTOD[3]){ EnableICBit(cia, 2); } } function TODLatch(cia){ if (cia.__TODLatch === null){ cia.__TODLatch = [ cia.__TOD[0], cia.__TOD[1], cia.__TOD[2], cia.__TOD[3] ]; } } function TimerATick(cia){ if (cia.__timerA > 0){ cia.__timerA -= 1; } else { // Check if Timer A is continuous if (BITM.val(cia.__CTA, 3) === 0){ cia.__timerA = cia.__timerALatch; } else { // Not continuous, so clear bit 0 (stop timer). cia.__CTA = BITM.clear(cia.__CTA, 0); } // Setting the interrupt bit for timer A EnableICBit(cia, 0); //cia.__IC = BITM.set(cia.__IC, 0); // Check to see if timer modifies the PORT B bit 6. // If so, determine if it's a pulse or an toggle. if (BITM.isOn(cia.__CTA, 1)) cia.__PDB = (BITM.isOn(cia.__CTA, 2) === 0) ? BITM.set(cia.__PDB, 6) : BITM.toggle(cia.__PDB, 6); // If timer B is running, and timer B tracks timer A underflows // tick timer B (or, tick it if CNT is positive) if (BITM.isOn(cia.__CTB, 0)){ let cs = (this.__CTB & 0x60) >> 5; if (cs === 2 || (cs === 3 && cia.__CNT === 1)) TimerBTick(cia); } } } function TimerBTick(cia){ if (cia.__timerB > 0){ cia.__timerB -= 1; } else { if (BITM.val(cia.__CTB, 3) === 0){ cia.__timerB = cia.__timerBLatch; } else { cia.__CTB = BITM.clear(cia.__CTB, 0); } EnableICBit(cia, 1); //cia.__IC = BITM.set(cia.__IC, 1); if (BITM.isOn(cia.__CTB, 1)) cia.__PDB = (BITM.isOn(cia.__CTB, 2) === 0) ? BITM.set(cia.__PDB, 7) : BITM.toggle(cia.__PDB, 7); } } function Tick(cia){ if (BITM.isOn(cia.__CTA, 0)){ if (BITM.isOn(cia.__CTA, 1) && !BITM.isOn(cia.__CTA, 2)) cia.__PDB = BITM.clear(cia.__PDB, 6); if (!BITM.isOn(cia.__CTA, 5)){ TimerATick(cia); } } if (BITM.isOn(cia.__CTB, 0)){ if (BITM.isOn(cia.__CTB, 1) && !BITM.isOn(cia.__CTB, 2)) cia.__PDB = BITM.clear(cia.__PDB, 7); let cs = (cia.__CTB & 0x60) >> 5; if (cs === 0) TimerBTick(cia); } if (cia.__CNT === 1){ if (BITM.isOn(cia.__CTA, 0) && BITM.isOn(cia.__CTA, 5)) TimerATick(cia); if (BITM.isOn(cia.__CTB, 0)){ let cs = (cia.__CTB & 0x60) >> 5; if (cs === 1) TimerBTick(cia); } if (BITM.isOn(cia.__CTA, 6)){ if (cia.__SPi > 0){ cia.__SP = BITM.val(cia.__SData, 7); cia.__SData = (cia.__SData << 1) & 0xFF; cia.__SPi -= 1; if (cia.__SPi === 0) EnableICBit(cia, 3); //cia.__IC = BITM.set(cia.__IC, 3); } } else { if (cia.__SPi < 8){ cia.__SData = (cia.__SData << 1) | cia.__SP; cia.__SPi += 1; if (cia.__SPi === 8) EnableICBit(cia, 3); //cia.__IC = BITM.set(cia.__IC, 3); } } } cia.__CNT = 0; } class MOSCIA{ constructor(){ this.__io = new IO(["PDA", "PDB"], ["PDA", "PDB"]); this.__regsel = 0; this.__RW = 0; this.__CNT = 0; this.__SP = 0; this.__SPi = 0; this.__SData = 0; this.__PDA = 0; this.__PDB = 0; this.__DDA = 0; this.__DDB = 0; this.__CTA = 0; this.__CTB = 0; this.__IC = 0; this.__ICMask = 0; this.__timerA = 0; this.__timerALatch = 1; this.__timerB = 0; this.__timerBLatch = 1; // Time Of Day this.__TODTick = 0; this.__TODLatch = null; this.__TOD = [0,0,0,0]; this.__ALTOD = [0,0,0,0]; } get RS(){return this.__regsel;} set RS(rs){this.__regsel = rs & 0x0F;} get RW(){return this.__RW;} set RW(rw){this.__RW = (rw === true || rw === 1) ? 1 : 0;} get DATA(){ let val = 0; switch(this.__regsel){ case 0x00: // Peripheral Data A this.__io.triggerRead("PDA", this.__DDA); val = this.__PDA; break; case 0x01: // Peripheral Data B this.__io.triggerRead("PDB", this.__DDB); val = this.__PDB; break; case 0x02: // Data Direction Reg A val = this.__DDA; break; case 0x03: // Data Direction Reg B val = this.__DDB; break; case 0x04: // Timer A Low val = (this.__timerA & 0x00FF); break; case 0x05: // Timer A High val = ((this.__timerA & 0xFF00) >> 8); break; case 0x06: // Timer B Low val = (this.__timerB & 0x00FF); break; case 0x07: // Timer B High val = ((this.__timerB & 0xFF00) >> 8); break; case 0x08: // 10th of Sec Reg val = (this.__TODLatch !== null) ? this.__TODLatch[0] : this.__TOD[0]; this.__TODLatch = null; break; case 0x09: // Seconds Reg val = (this.__TODLatch !== null) ? this.__TODLatch[1] : this.__TOD[1]; break; case 0x0A: // Minutes Reg val = (this.__TODLatch !== null) ? this.__TODLatch[2] : this.__TOD[2]; break; case 0x0B: // Hours AM/PM Reg TODLatch(this); val = this.__TODLatch[3]; break; case 0x0C: // Serial Data val = this.__SData; break; case 0x0D: // Interrupt Control let v = this.__IC; this.__IC = 0; return v; break; case 0x0E: // Control Timer Reg A val = this.__CTA; break; case 0x0F: // Control Timer Reg B val = this.__CTB; break; } return val; } set DATA(d){ if (this.__RW === 1){return;} let tod = 0; switch(this.__regsel){ case 0x00: // Peripheral Data A this.__io.triggerWrite("PDA", this.__PDA, d & 0xFF); this.__PDA = (this.__PDA & (~this.__DDA)) | ((d & 0xFF) & this.__DDA); break; case 0x01: // Peripheral Data B this.__io.triggerWrite("PDB", this.__PDB, d & 0xFF); this.__PDB = (this.__PDB & (~this.__DDB)) | ((d & 0xFF) & this.__DDB); break; case 0x02: // Data Direction Reg A this.__DDA = d & 0xFF; break; case 0x03: // Data Direction Reg B this.__DDB = d & 0xFF; break; case 0x04: // Timer A Low this.__timerALatch = (this.__timerALatch & 0xFF00) | (d & 0xFF); break; case 0x05: // Timer A High this.__timerALatch = (this.__timerALatch & 0x00FF) | ((d & 0xFF) << 8); break case 0x06: // Timer B Low this.__timerBLatch = (this.__timerBLatch & 0xFF00) | (d & 0xFF); break; case 0x07: // Timer B High this.__timerBLatch = (this.__timerBLatch & 0x00FF) | ((d & 0xFF) << 8); break; case 0x08: // 10th of Sec Reg if (this.__TODTick < 0){ tod = (BITM.isOn(this.__CTB, 7)) ? this.__ALTOD : this.__TOD; tod[0] = d & 0xFF; this.__TODTick = (this.__CTA & 0x80) >> 7; // Restarts TOD cycling. } break; case 0x09: // Seconds Reg if (this.__TODTick < 0){ tod = (BITM.isOn(this.__CTB, 7)) ? this.__ALTOD : this.__TOD; tod[1] = d & 0xFF; } break; case 0x0A: // Minutes Reg if (this.__TODTick < 0){ tod = (BITM.isOn(this.__CTB, 7)) ? this.__ALTOD : this.__TOD; tod[2] = d & 0xFF; } break; case 0x0B: // Hours AM/PM Reg this.__TODTick = -1; // This will prevent TOD from cycling. tod = (BITM.isOn(this.__CTB, 7)) ? this.__ALTOD : this.__TOD; tod[3] = d & 0xFF; break; case 0x0C: // Serial Data break; case 0x0D: // Interrupt Control if (BITM.isOn(d & 0xFF, 7)){ for (let i=0; i < 5; i++){ if (BITM.isOn(d & 0xFF, i)) this.__ICMask = BITM.set(this.__ICMask, i); } } else { for (let i=0; i < 5; i++){ if (BITM.isOn(d & 0xFF, i)) this.__ICMask = BITM.clear(this.__ICMask, i); } } break; case 0x0E: // Control Timer Reg A this.__CTA = d & 0xFF; if (BITM.isOn(this.__CTA, 4)) this.__timerA = this.__timerALatch; // If __TODTick is less than 0, TOD is in write mode. // As such, the __TODTick counter will be reset once 10ths is // written to... so there's no need to adjust the ticks here... // Besides, doing so will break the system. if (this.__TODTick >= 0){ if (BITM.isOn(this.__CTA, 7)){ this.__TODTick = (this.__TODTick === 5) ? 0 : this.__TODTick + 1; } else { this.__TODTick -= (this.__TODTick === 5) ? this.__TODTick : 0; } } break; case 0x0F: // Control Timer Reg B this.__CTB = d & 0xFF; if (BITM.isOn(this.__CTB, 4)) this.__timerB = this.__timerBLatch; break; } } get CNT(){return this.__CNT;} set CNT(c){ this.__CNT = (c >= 1) ? 1 : 0; } get SP(){return this.__SP;} set SP(sp){ this.__SP = (sp === true || sp === 1) ? 1 : 0; } set FLAG(f){ if (f === true || f === 1) EnableICBit(this, 4); } get TOD(){return 0;} set TOD(b){ if (b > 0 || this.__TODLatch < 2) TODTick(this); } get phi2(){return 0;} set phi2(p){ if (p === true || p === 1){ Tick(this); } } onRead(name, fn){ this.__io.onRead(name, fn); return this; } onWrite(name, fn){ this.__io.onWrite(name, fn); return this; } setPDA(v){ this.__PDA = (this.__PDA & this.__DDA) | ((v & 0xFF) & (~this.__DDA)); return this; } setPDB(v){ this.__PDB = (this.__PDB & this.__DDB) | ((v & 0xFF) & (~this.__DDB)); return this; } setTOD(h,m,s,t){ // This method primarily exists for testing purposes. This should NOT // be used in a true emulation. this.__TOD[0] = t & 0xFF; this.__TOD[1] = s & 0xFF; this.__TOD[2] = m & 0xFF; this.__TOD[3] = h & 0xFF; } reset(){ this.__CNT = 0; this.__PDA = 0; this.__PDB = 0; this.__DDA = 0; this.__DDB = 0; this.__CTA = 0; this.__CTB = 0; this.__IC = 0; this.__ICMask = 0; this.__timerA = 0; this.__timerALatch = 1; this.__timerB = 0; this.__timerBLatch = 1; // Time Of Day this.__TODLatch = 0; this.__TODTick = 0; this.__TOD = [0,0,0,0]; this.__LTOD = [0,0,0,0]; this.__ALTOD = [0,0,0,0]; return this; } } module.exports = MOSCIA;