1
0
mirror of https://github.com/sehugg/8bitworkshop.git synced 2025-01-20 07:33:11 +00:00
8bitworkshop/src/machine/vdp_z80.ts

99 lines
2.4 KiB
TypeScript

import { Z80, Z80State } from "../common/cpu/ZilogZ80";
import { BasicScanlineMachine, Bus, ProbeAll } from "../common/devices";
import { newAddressDecoder, newKeyboardHandler } from "../common/emu";
import { TssChannelAdapter } from "../common/audio";
import { TMS9918A } from "../common/video/tms9918a";
const audioOversample = 2;
export abstract class BaseZ80VDPBasedMachine extends BasicScanlineMachine {
cpuFrequency = 3579545; // MHz
canvasWidth = 304;
numTotalScanlines = 262;
numVisibleScanlines = 240;
cpuCyclesPerLine = this.cpuFrequency / (262*60);
sampleRate = 262*60*audioOversample;
overscan = true;
cpu: Z80 = new Z80();
vdp: TMS9918A;
psg;
audioadapter;
abstract vdpInterrupt();
abstract getKeyboardMap();
getKeyboardFunction() { return null; }
init(membus:Bus, iobus:Bus, psg) {
this.connectCPUMemoryBus(membus);
this.connectCPUIOBus(iobus);
this.handler = newKeyboardHandler(this.inputs, this.getKeyboardMap(), this.getKeyboardFunction());
this.psg = psg;
this.audioadapter = psg && new TssChannelAdapter(psg.psg, audioOversample, this.sampleRate);
}
connectVideo(pixels) {
super.connectVideo(pixels);
var cru = {
setVDPInterrupt: (b) => {
if (b) {
this.vdpInterrupt();
} else {
// TODO: reset interrupt?
}
}
};
this.vdp = this.newVDP(this.pixels, cru, true);
}
connectProbe(probe: ProbeAll) : void {
super.connectProbe(probe);
this.vdp.probe = probe || this.nullProbe;
}
newVDP(frameData, cru, flicker) {
return new TMS9918A(frameData, cru, flicker);
}
startScanline() {
this.audio && this.audioadapter && this.audioadapter.generate(this.audio);
}
drawScanline() {
this.vdp.drawScanline(this.scanline);
}
loadState(state) {
super.loadState(state);
this.vdp.restoreState(state['vdp']);
}
saveState() {
var state = super.saveState();
state['vdp'] = this.vdp.getState();
return state;
}
reset() {
super.reset();
this.vdp.reset();
this.psg.reset();
}
getDebugCategories() {
return ['CPU','Stack','VDP'];
}
getDebugInfo(category, state) {
switch (category) {
case 'VDP': return this.vdpStateToLongString(state.vdp);
}
}
vdpStateToLongString(ppu) {
return this.vdp.getRegsString();
}
readVRAMAddress(a : number) : number {
return this.vdp.ram[a & 0x3fff];
}
}