1
0
mirror of https://github.com/sehugg/8bitworkshop.git synced 2025-07-19 10:24:24 +00:00

updated coleco to BasicZ80Scanline

This commit is contained in:
Steven Hugg
2019-08-04 20:10:14 -04:00
parent 221d16a3c0
commit 725d0c9efa
3 changed files with 161 additions and 188 deletions

View File

@@ -52,6 +52,7 @@ const _Apple2Platform = function(mainElement) {
// bank 1 is E000-FFFF, bank 2 is D000-DFFF // bank 1 is E000-FFFF, bank 2 is D000-DFFF
var bank2rdoffset=0, bank2wroffset=0; var bank2rdoffset=0, bank2wroffset=0;
var grparams : AppleGRParams; var grparams : AppleGRParams;
var scanline : number;
class Apple2Platform extends Base6502Platform implements Platform { class Apple2Platform extends Base6502Platform implements Platform {
@@ -199,6 +200,7 @@ const _Apple2Platform = function(mainElement) {
var clock = 0; var clock = 0;
var debugCond = this.getDebugCallback(); var debugCond = this.getDebugCallback();
for (var sl=0; sl<262; sl++) { for (var sl=0; sl<262; sl++) {
scanline = sl;
for (var i=0; i<cpuCyclesPerLine; i++) { for (var i=0; i<cpuCyclesPerLine; i++) {
if (debugCond && debugCond()) { if (debugCond && debugCond()) {
debugCond = null; debugCond = null;
@@ -218,6 +220,10 @@ const _Apple2Platform = function(mainElement) {
} }
} }
getRasterScanline() : number {
return scanline;
}
loadROM(title, data) { loadROM(title, data) {
pgmbin = data; pgmbin = data;
this.reset(); this.reset();

View File

@@ -1,7 +1,7 @@
"use strict"; "use strict";
import { Platform, BaseMAMEPlatform, BaseZ80Platform, getToolForFilename_z80 } from "../baseplatform"; import { Platform, BaseMAMEPlatform, BasicZ80ScanlinePlatform, getToolForFilename_z80 } from "../baseplatform";
import { PLATFORMS, RAM, newAddressDecoder, padBytes, noise, setKeyboardFromMap, AnimationTimer, RasterVideo, Keys, makeKeycodeMap } from "../emu"; import { PLATFORMS, newAddressDecoder, padBytes, noise, setKeyboardFromMap, AnimationTimer, RasterVideo, Keys, makeKeycodeMap } from "../emu";
import { hex, lzgmini, stringToByteArray } from "../util"; import { hex, lzgmini, stringToByteArray } from "../util";
import { MasterAudio, SN76489_Audio } from "../audio"; import { MasterAudio, SN76489_Audio } from "../audio";
import { TMS9918A } from "../video/tms9918a"; import { TMS9918A } from "../video/tms9918a";
@@ -25,20 +25,20 @@ import { TMS9918A } from "../video/tms9918a";
// http://www.harmlesslion.com/cgi-bin/showprog.cgi?ColecoVision // http://www.harmlesslion.com/cgi-bin/showprog.cgi?ColecoVision
export var ColecoVision_PRESETS = [ export var ColecoVision_PRESETS = [
{id:'text.c', name:'Text Mode'}, { id: 'text.c', name: 'Text Mode' },
{id:'hello.c', name:'Scrolling Text'}, { id: 'hello.c', name: 'Scrolling Text' },
{id:'text32.c', name:'32-Column Color Text'}, { id: 'text32.c', name: '32-Column Color Text' },
{id:'stars.c', name:'Scrolling Starfield'}, { id: 'stars.c', name: 'Scrolling Starfield' },
{id:'cursorsmooth.c', name:'Moving Cursor'}, { id: 'cursorsmooth.c', name: 'Moving Cursor' },
{id:'simplemusic.c', name:'Simple Music'}, { id: 'simplemusic.c', name: 'Simple Music' },
{id:'musicplayer.c', name:'Multivoice Music'}, { id: 'musicplayer.c', name: 'Multivoice Music' },
{id:'mode2bitmap.c', name:'Mode 2 Bitmap'}, { id: 'mode2bitmap.c', name: 'Mode 2 Bitmap' },
{id:'mode2compressed.c', name:'Mode 2 Bitmap (LZG)'}, { id: 'mode2compressed.c', name: 'Mode 2 Bitmap (LZG)' },
{id:'lines.c', name:'Mode 2 Lines'}, { id: 'lines.c', name: 'Mode 2 Lines' },
{id:'multicolor.c', name:'Multicolor Mode'}, { id: 'multicolor.c', name: 'Multicolor Mode' },
{id:'siegegame.c', name:'Siege Game'}, { id: 'siegegame.c', name: 'Siege Game' },
{id:'shoot.c', name:'Solarian Game'}, { id: 'shoot.c', name: 'Solarian Game' },
{id:'climber.c', name:'Platform Game'}, { id: 'climber.c', name: 'Platform Game' },
]; ];
var COLECOVISION_KEYCODE_MAP = makeKeycodeMap([ var COLECOVISION_KEYCODE_MAP = makeKeycodeMap([
@@ -57,154 +57,122 @@ var COLECOVISION_KEYCODE_MAP = makeKeycodeMap([
[Keys.P2_B, 3, 0x40], [Keys.P2_B, 3, 0x40],
]); ]);
/// standard emulator class ColecoVisionPlatform extends BasicZ80ScanlinePlatform implements Platform {
const _ColecoVisionPlatform = function(mainElement) { cpuFrequency = 3579545; // MHz
canvasWidth = 304;
numTotalScanlines = 262;
numVisibleScanlines = 240;
defaultROMSize = 0x8000;
const cpuFrequency = 3579545; // MHz vdp: TMS9918A;
const canvasWidth = 304; bios: Uint8Array;
const numTotalScanlines = 262; keypadMode: boolean;
const numVisibleScanlines = 240; audio;
const cpuCyclesPerLine = Math.round(cpuFrequency / 60 / numTotalScanlines); psg;
var cpu, ram, membus, iobus, rom, bios;
var video, vdp, timer;
var audio, psg;
var inputs = new Uint8Array(4);
var keypadMode = false;
var poller;
class ColecoVisionPlatform extends BaseZ80Platform implements Platform {
getPresets() { return ColecoVision_PRESETS; } getPresets() { return ColecoVision_PRESETS; }
start() { getKeyboardMap() { return COLECOVISION_KEYCODE_MAP; }
ram = new RAM(1024);
bios = new lzgmini().decode(stringToByteArray(atob(COLECO_BIOS_LZG))); getVideoOptions() { return { overscan: true }; }
membus = {
newRAM() {
return new Uint8Array(0x400);
}
newMembus() {
return {
read: newAddressDecoder([ read: newAddressDecoder([
[0x0000, 0x1fff, 0x1fff, function(a) { return bios ? bios[a] : 0; }], [0x0000, 0x1fff, 0x1fff, (a) => { return this.bios ? this.bios[a] : 0; }],
[0x6000, 0x7fff, 0x3ff, function(a) { return ram.mem[a]; }], [0x6000, 0x7fff, 0x3ff, (a) => { return this.ram[a]; }],
[0x8000, 0xffff, 0x7fff, function(a) { return rom ? rom[a] : 0; }], [0x8000, 0xffff, 0x7fff, (a) => { return this.rom ? this.rom[a] : 0; }],
]), ]),
write: newAddressDecoder([ write: newAddressDecoder([
[0x6000, 0x7fff, 0x3ff, function(a,v) { ram.mem[a] = v; }], [0x6000, 0x7fff, 0x3ff, (a, v) => { this.ram[a] = v; }],
]), ]),
}; };
iobus = { }
read: function(addr) {
newIOBus() {
return {
read: (addr:number):number => {
addr &= 0xff; addr &= 0xff;
//console.log('IO read', hex(addr,4)); //console.log('IO read', hex(addr,4));
switch (addr) { switch (addr) {
case 0xfc: return inputs[keypadMode?1:0] ^ 0xff; case 0xfc: return this.inputs[this.keypadMode ? 1 : 0] ^ 0xff;
case 0xff: return inputs[keypadMode?3:2] ^ 0xff; case 0xff: return this.inputs[this.keypadMode ? 3 : 2] ^ 0xff;
} }
if (addr >= 0xa0 && addr <= 0xbf) { if (addr >= 0xa0 && addr <= 0xbf) {
if (addr & 1) if (addr & 1)
return vdp.readStatus(); return this.vdp.readStatus();
else else
return vdp.readData(); return this.vdp.readData();
} }
return 0; return 0;
}, },
write: function(addr, val) { write: (addr:number, val:number) => {
addr &= 0xff; addr &= 0xff;
val &= 0xff; val &= 0xff;
//console.log('IO write', hex(addr,4), hex(val,2)); //console.log('IO write', hex(addr,4), hex(val,2));
switch (addr >> 4) { switch (addr >> 4) {
case 0x8: case 0x9: keypadMode = true; break; case 0x8: case 0x9: this.keypadMode = true; break;
case 0xc: case 0xd: keypadMode = false; break; case 0xc: case 0xd: this.keypadMode = false; break;
case 0xa: case 0xb: case 0xa: case 0xb:
if (addr & 1) if (addr & 1)
return vdp.writeAddress(val); return this.vdp.writeAddress(val);
else else
return vdp.writeData(val); return this.vdp.writeData(val);
case 0xf: psg.setData(val); break; case 0xf: this.psg.setData(val); break;
} }
} }
}; };
cpu = this.newCPU(membus, iobus); }
video = new RasterVideo(mainElement,canvasWidth,numVisibleScanlines,{overscan:true});
video.create(); start() {
audio = new MasterAudio(); super.start();
psg = new SN76489_Audio(audio); this.bios = new lzgmini().decode(stringToByteArray(atob(COLECO_BIOS_LZG)));
this.audio = new MasterAudio();
this.psg = new SN76489_Audio(this.audio);
var cru = { var cru = {
setVDPInterrupt: (b) => { setVDPInterrupt: (b) => {
if (b) { if (b) {
cpu.nonMaskableInterrupt(); this.cpu.nonMaskableInterrupt();
} else { } else {
// TODO: reset interrupt? // TODO: reset interrupt?
} }
} }
}; };
vdp = new TMS9918A(video.getFrameData(), cru, true); // true = 4 sprites/line this.vdp = this.newVDP(this.video.getFrameData(), cru, true); // true = 4 sprites/line
poller = setKeyboardFromMap(video, inputs, COLECOVISION_KEYCODE_MAP);
timer = new AnimationTimer(60, this.nextFrame.bind(this));
} }
pollControls() { poller.poll(); } newVDP(frameData, cru, flicker) {
return new TMS9918A(frameData, cru, flicker);
readAddress(addr) {
return membus.read(addr);
} }
advance(novideo : boolean) { startScanline(sl: number) {
for (var sl=0; sl<numTotalScanlines; sl++) {
this.runCPU(cpu, cpuCyclesPerLine);
vdp.drawScanline(sl);
}
video.updateFrame();
} }
loadROM(title, data) { drawScanline(sl: number) {
rom = padBytes(data, 0x8000); this.vdp.drawScanline(sl);
this.reset();
} }
loadState(state) { loadState(state) {
cpu.loadState(state.c); super.loadState(state);
ram.mem.set(state.b); this.vdp.restoreState(state['vdp']);
vdp.restoreState(state.vdp); this.keypadMode = state['kpm'];
keypadMode = state.kpm;
inputs.set(state.in);
} }
saveState() { saveState() {
return { var state = super.saveState();
c:this.getCPUState(), state['vdp'] = this.vdp.getState();
b:ram.mem.slice(0), state['kpm'] = this.keypadMode;
vdp:vdp.getState(), return state;
kpm:keypadMode,
in:inputs.slice(0),
};
}
loadControlsState(state) {
inputs.set(state.in);
}
saveControlsState() {
return {
in:inputs.slice(0)
};
}
getCPUState() {
return cpu.saveState();
}
isRunning() {
return timer && timer.isRunning();
}
pause() {
timer.stop();
audio.stop();
}
resume() {
timer.start();
audio.start();
} }
reset() { reset() {
cpu.reset(); super.reset();
cpu.setTstates(0); this.vdp.reset();
vdp.reset(); this.psg.reset();
psg.reset(); this.keypadMode = false;
} }
getDebugCategories() { getDebugCategories() {
@@ -217,10 +185,8 @@ const _ColecoVisionPlatform = function(mainElement) {
} }
} }
vdpStateToLongString(ppu) { vdpStateToLongString(ppu) {
return vdp.getRegsString(); return this.vdp.getRegsString();
} }
}
return new ColecoVisionPlatform();
} }
var COLECO_BIOS_LZG = ` var COLECO_BIOS_LZG = `
@@ -266,15 +232,15 @@ class ColecoVisionMAMEPlatform extends BaseMAMEPlatform implements Platform {
start() { start() {
this.startModule(this.mainElement, { this.startModule(this.mainElement, {
jsfile:'mamecoleco.js', jsfile: 'mamecoleco.js',
cfgfile:'coleco.cfg', cfgfile: 'coleco.cfg',
biosfile:'coleco/313 10031-4005 73108a.u2', biosfile: 'coleco/313 10031-4005 73108a.u2',
driver:'coleco', driver: 'coleco',
width:280*2, width: 280 * 2,
height:216*2, height: 216 * 2,
romfn:'/emulator/cart.rom', romfn: '/emulator/cart.rom',
romsize:0x8000, romsize: 0x8000,
preInit:function(_self) { preInit: function(_self) {
}, },
}); });
} }
@@ -294,4 +260,4 @@ class ColecoVisionMAMEPlatform extends BaseMAMEPlatform implements Platform {
/// ///
PLATFORMS['coleco.mame'] = ColecoVisionMAMEPlatform; PLATFORMS['coleco.mame'] = ColecoVisionMAMEPlatform;
PLATFORMS['coleco'] = _ColecoVisionPlatform; PLATFORMS['coleco'] = ColecoVisionPlatform;

View File

@@ -12,6 +12,7 @@ import { TMS9918A } from "../video/tms9918a";
// https://openmsx.org/manual/setup.html // https://openmsx.org/manual/setup.html
// https://www.msx.org/wiki/Slots // https://www.msx.org/wiki/Slots
// https://www.msx.org/wiki/SDCC // https://www.msx.org/wiki/SDCC
// http://cbios.sourceforge.net/
var MSX_PRESETS = [ var MSX_PRESETS = [
{id:'helloworld.asm', name:'Hello World (ASM)'}, {id:'helloworld.asm', name:'Hello World (ASM)'},