소스 검색

Fixed TOD issues. Working on interrupt issues.

master
Bryan Miller 5 년 전
부모
커밋
de34b64ed7
1개의 변경된 파일70개의 추가작업 그리고 38개의 파일을 삭제
  1. +70
    -38
      src/MOSCIA.js

+ 70
- 38
src/MOSCIA.js 파일 보기

const BCD = require('./utils/bcd.js'); const BCD = require('./utils/bcd.js');
const BITM = require('./utils/bitman.js'); const BITM = require('./utils/bitman.js');


function ChangeICBit(cia, v, pos){
let mask = ~(1 << pos);
v = ((v === true || v === 1) ? 1 : 0) << pos;
cia.__IC = (cia.__IC & mask) | v;
function EnableICBit(cia, pos){
v = 1 << pos;
cia.__IC = cia.__IC | v;
// Enabling any bit enables bit 7 as well!
cia.__IC = cia.__IC | 0x80;
} }




function TODTick(cia){ function TODTick(cia){
if (cia.__TODTick < 0){return;}
cia.__TODTick += 1; cia.__TODTick += 1;
if (cia.__TODTick === 6){ if (cia.__TODTick === 6){
cia.__TODTick = cia.__CTA & 0x01;
cia.__TODTick = (cia.__CTA & 0x80) >> 7;
if (cia.__TOD[0] == 0x09){ if (cia.__TOD[0] == 0x09){
cia.__TOD[0] = 0; cia.__TOD[0] = 0;
if (cia.__TOD[1] == 0x59){ if (cia.__TOD[1] == 0x59){
if (cia.__TOD[2] == 0x59){ if (cia.__TOD[2] == 0x59){
cia.__TOD[2] = 0; cia.__TOD[2] = 0;
if (cia.__TOD[3] == 0x11 || cia.__TOD[3] == 0x91){ if (cia.__TOD[3] == 0x11 || cia.__TOD[3] == 0x91){
cia.__TOD[3] = (cia.__TOD[3] == 0x91) ? 0x00 : 0x10;
cia.__TOD[3] = (cia.__TOD[3] == 0x91) ? 0x00 : 0x80;
} else { } else {
let pm = cia.__TOD[3] & 0x80; let pm = cia.__TOD[3] & 0x80;
cia.__TOD[3] = BCD.add(cia.__TOD[3] & 0x7F, 0x01, 2) | pm; cia.__TOD[3] = BCD.add(cia.__TOD[3] & 0x7F, 0x01, 2) | pm;
} }


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]){ 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]){
ChangeICBit(cia, 1, 2);
EnableICBit(cia, 2);
} }
} }


function TODLatch(cia){ function TODLatch(cia){
if (cia.__TODLatch === 0){
cia.__TODLatch = 1;
cia.__LTOD[0] = cia.__TOD[0];
cia.__LTOD[1] = cia.__TOD[1];
cia.__LTOD[2] = cia.__TOD[2];
cia.__LTOD[3] = cia.__TOD[3];
if (cia.__TODLatch === null){
cia.__TODLatch = [
cia.__TOD[0],
cia.__TOD[1],
cia.__TOD[2],
cia.__TOD[3]
];
} }
} }


} }


// Setting the interrupt bit for timer A // Setting the interrupt bit for timer A
cia.__IC = BITM.set(cia.__IC, 0);
EnableICBit(cia, 0);
//cia.__IC = BITM.set(cia.__IC, 0);


// Check to see if timer modifies the PORT B bit 6. // Check to see if timer modifies the PORT B bit 6.
// If so, determine if it's a pulse or an toggle. // If so, determine if it's a pulse or an toggle.
cia.__CTB = BITM.clear(cia.__CTB, 0); cia.__CTB = BITM.clear(cia.__CTB, 0);
} }


cia.__IC = BITM.set(cia.__IC, 1);
EnableICBit(cia, 1);
//cia.__IC = BITM.set(cia.__IC, 1);


if (BITM.isOn(cia.__CTB, 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); cia.__PDB = (BITM.isOn(cia.__CTB, 2) === 0) ? BITM.set(cia.__PDB, 7) : BITM.toggle(cia.__PDB, 7);
cia.__SData = (cia.__SData << 1) & 0xFF; cia.__SData = (cia.__SData << 1) & 0xFF;
cia.__SPi -= 1; cia.__SPi -= 1;
if (cia.__SPi === 0) if (cia.__SPi === 0)
cia.__IC = BITM.set(cia.__IC, 3);
EnableICBit(cia, 3);
//cia.__IC = BITM.set(cia.__IC, 3);
} }
} else { } else {
if (cia.__SPi < 8){ if (cia.__SPi < 8){
cia.__SData = (cia.__SData << 1) | cia.__SP; cia.__SData = (cia.__SData << 1) | cia.__SP;
cia.__SPi += 1; cia.__SPi += 1;
if (cia.__SPi === 8) if (cia.__SPi === 8)
cia.__IC = BITM.set(cia.__IC, 3);
EnableICBit(cia, 3);
//cia.__IC = BITM.set(cia.__IC, 3);
} }
} }
} }
this.__timerBLatch = 1; this.__timerBLatch = 1;


// Time Of Day // Time Of Day
this.__TODLatch = 0;
this.__TODTick = 0; this.__TODTick = 0;
this.__TODLatch = null;
this.__TOD = [0,0,0,0]; this.__TOD = [0,0,0,0];
this.__LTOD = [0,0,0,0];
this.__ALTOD = [0,0,0,0]; this.__ALTOD = [0,0,0,0];
} }


case 0x07: // Timer B High case 0x07: // Timer B High
val = ((this.__timerB & 0xFF00) >> 4); break; val = ((this.__timerB & 0xFF00) >> 4); break;
case 0x08: // 10th of Sec Reg case 0x08: // 10th of Sec Reg
this.__TODLatch = 0;
val = this.__TOD[0]; break;
val = (this.__TODLatch !== null) ? this.__TODLatch[0] : this.__TOD[0];
this.__TODLatch = null;
break;
case 0x09: // Seconds Reg case 0x09: // Seconds Reg
val = (this.__TODLatch > 0) ? this.__LTOD[1] : this.__TOD[1]; break;
val = (this.__TODLatch !== null) ? this.__TODLatch[1] : this.__TOD[1]; break;
case 0x0A: // Minutes Reg case 0x0A: // Minutes Reg
val = (this.__TODLatch > 0) ? this.__LTOD[2] : this.__TOD[2]; break;
val = (this.__TODLatch !== null) ? this.__TODLatch[2] : this.__TOD[2]; break;
case 0x0B: // Hours AM/PM Reg case 0x0B: // Hours AM/PM Reg
TODLatch(this); TODLatch(this);
val = this.__LTOD[3]; break;
val = this.__TODLatch[3]; break;
case 0x0C: // Serial Data case 0x0C: // Serial Data
val = this.__SData; break; val = this.__SData; break;
case 0x0D: // Interrupt Control case 0x0D: // Interrupt Control
this.__timerBLatch = (this.__timerBLatch & 0x00FF) | ((d & 0xFF) << 4); this.__timerBLatch = (this.__timerBLatch & 0x00FF) | ((d & 0xFF) << 4);
break; break;
case 0x08: // 10th of Sec Reg case 0x08: // 10th of Sec Reg
TODLatch(this);
this.__TODLatch = 2;
tod = (BITM.isOn(this.__CTB, 7)) ? this.__ALTOD : this.__TOD;
tod[0] = d & 0xFF;
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; break;
case 0x09: // Seconds Reg case 0x09: // Seconds Reg
TODLatch(this);
this.__TODLatch = 2;
tod = (BITM.isOn(this.__CTB, 7)) ? this.__ALTOD : this.__TOD;
tod[1] = d & 0xFF;
if (this.__TODTick < 0){
tod = (BITM.isOn(this.__CTB, 7)) ? this.__ALTOD : this.__TOD;
tod[1] = d & 0xFF;
}
break; break;
case 0x0A: // Minutes Reg case 0x0A: // Minutes Reg
TODLatch(this);
this.__TODLatch = 2;
tod = (BITM.isOn(this.__CTB, 7)) ? this.__ALTOD : this.__TOD;
tod[2] = d & 0xFF;
if (this.__TODTick < 0){
tod = (BITM.isOn(this.__CTB, 7)) ? this.__ALTOD : this.__TOD;
tod[2] = d & 0xFF;
}
break; break;
case 0x0B: // Hours AM/PM Reg case 0x0B: // Hours AM/PM Reg
TODLatch(this);
this.__TODLatch = 2;
this.__TODTick = -1; // This will prevent TOD from cycling.
tod = (BITM.isOn(this.__CTB, 7)) ? this.__ALTOD : this.__TOD; tod = (BITM.isOn(this.__CTB, 7)) ? this.__ALTOD : this.__TOD;
tod[3] = d & 0xFF; tod[3] = d & 0xFF;
break; break;
this.__CTA = d & 0xFF; this.__CTA = d & 0xFF;
if (BITM.isOn(this.__CTA, 4)) if (BITM.isOn(this.__CTA, 4))
this.__timerA = this.__timerALatch; 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; break;
case 0x0F: // Control Timer Reg B case 0x0F: // Control Timer Reg B
this.__CTB = d & 0xFF; this.__CTB = d & 0xFF;
this.__SP = (sp === true || sp === 1) ? 1 : 0; this.__SP = (sp === true || sp === 1) ? 1 : 0;
} }


set FLAG(f){
if (f === true || f === 1)
EnableICBit(this, 4);
}

get TOD(){return 0;} get TOD(){return 0;}
set TOD(b){ set TOD(b){
if (b > 0 || this.__TODLatch < 2) if (b > 0 || this.__TODLatch < 2)
return this; 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(){ reset(){
this.__CNT = 0; this.__CNT = 0;



Loading…
취소
저장