2018-08-16 23:19:20 +00:00
|
|
|
"use strict";
|
|
|
|
|
2019-08-05 00:10:14 +00:00
|
|
|
import { Platform, BaseMAMEPlatform, BasicZ80ScanlinePlatform, getToolForFilename_z80 } from "../baseplatform";
|
|
|
|
import { PLATFORMS, newAddressDecoder, padBytes, noise, setKeyboardFromMap, AnimationTimer, RasterVideo, Keys, makeKeycodeMap } from "../emu";
|
2018-11-19 14:10:13 +00:00
|
|
|
import { hex, lzgmini, stringToByteArray } from "../util";
|
|
|
|
import { MasterAudio, SN76489_Audio } from "../audio";
|
|
|
|
import { TMS9918A } from "../video/tms9918a";
|
2018-08-16 23:19:20 +00:00
|
|
|
|
|
|
|
// http://www.colecovision.eu/ColecoVision/development/tutorial1.shtml
|
|
|
|
// http://www.colecovision.eu/ColecoVision/development/libcv.shtml
|
|
|
|
// http://www.kernelcrash.com/blog/recreating-the-colecovision/2016/01/27/
|
|
|
|
// http://www.atarihq.com/danb/files/CV-Tech.txt
|
|
|
|
// http://www.atarihq.com/danb/files/CV-Sound.txt
|
|
|
|
// http://www.colecoboxart.com/faq/FAQ05.htm
|
|
|
|
// http://www.theadamresource.com/manuals/technical/Jeffcoleco.html
|
|
|
|
// http://bifi.msxnet.org/msxnet//tech/tms9918a.txt
|
|
|
|
// http://www.colecovision.dk/tools.htm?refreshed
|
|
|
|
// http://www.theadamresource.com/manuals/technical/ColecoVision%20Coding%20Guide.pdf
|
|
|
|
// http://www.unige.ch/medecine/nouspikel/ti99/tms9918a.htm
|
2018-11-22 21:32:55 +00:00
|
|
|
// http://map.grauw.nl/articles/vdp_tut.php
|
|
|
|
// http://www.msxcomputermagazine.nl/mccw/91/msx1demos1/en.html
|
|
|
|
// http://www.segordon.com/colecovision.php
|
|
|
|
// http://samdal.com/svvideo.htm
|
2018-11-26 12:33:09 +00:00
|
|
|
// https://github.com/tursilion/convert9918
|
|
|
|
// http://www.harmlesslion.com/cgi-bin/showprog.cgi?ColecoVision
|
2018-08-16 23:19:20 +00:00
|
|
|
|
2018-11-28 22:47:39 +00:00
|
|
|
export var ColecoVision_PRESETS = [
|
2019-08-05 00:10:14 +00:00
|
|
|
{ id: 'text.c', name: 'Text Mode' },
|
|
|
|
{ id: 'hello.c', name: 'Scrolling Text' },
|
|
|
|
{ id: 'text32.c', name: '32-Column Color Text' },
|
|
|
|
{ id: 'stars.c', name: 'Scrolling Starfield' },
|
|
|
|
{ id: 'cursorsmooth.c', name: 'Moving Cursor' },
|
|
|
|
{ id: 'simplemusic.c', name: 'Simple Music' },
|
|
|
|
{ id: 'musicplayer.c', name: 'Multivoice Music' },
|
|
|
|
{ id: 'mode2bitmap.c', name: 'Mode 2 Bitmap' },
|
|
|
|
{ id: 'mode2compressed.c', name: 'Mode 2 Bitmap (LZG)' },
|
|
|
|
{ id: 'lines.c', name: 'Mode 2 Lines' },
|
|
|
|
{ id: 'multicolor.c', name: 'Multicolor Mode' },
|
|
|
|
{ id: 'siegegame.c', name: 'Siege Game' },
|
|
|
|
{ id: 'shoot.c', name: 'Solarian Game' },
|
|
|
|
{ id: 'climber.c', name: 'Platform Game' },
|
2018-08-16 23:19:20 +00:00
|
|
|
];
|
|
|
|
|
2018-11-19 14:10:13 +00:00
|
|
|
var COLECOVISION_KEYCODE_MAP = makeKeycodeMap([
|
2019-08-05 00:10:14 +00:00
|
|
|
[Keys.UP, 0, 0x1],
|
|
|
|
[Keys.DOWN, 0, 0x4],
|
|
|
|
[Keys.LEFT, 0, 0x8],
|
2019-06-07 13:45:30 +00:00
|
|
|
[Keys.RIGHT, 0, 0x2],
|
2019-08-05 00:10:14 +00:00
|
|
|
[Keys.A, 0, 0x40],
|
|
|
|
[Keys.B, 1, 0x40],
|
2018-11-19 14:10:13 +00:00
|
|
|
|
2019-08-05 00:10:14 +00:00
|
|
|
[Keys.P2_UP, 2, 0x1],
|
|
|
|
[Keys.P2_DOWN, 2, 0x4],
|
|
|
|
[Keys.P2_LEFT, 2, 0x8],
|
2019-06-07 13:45:30 +00:00
|
|
|
[Keys.P2_RIGHT, 2, 0x2],
|
2019-08-05 00:10:14 +00:00
|
|
|
[Keys.P2_A, 2, 0x40],
|
|
|
|
[Keys.P2_B, 3, 0x40],
|
2018-11-19 14:10:13 +00:00
|
|
|
]);
|
|
|
|
|
2019-08-05 00:10:14 +00:00
|
|
|
class ColecoVisionPlatform extends BasicZ80ScanlinePlatform implements Platform {
|
2018-11-19 14:10:13 +00:00
|
|
|
|
2019-08-05 00:10:14 +00:00
|
|
|
cpuFrequency = 3579545; // MHz
|
|
|
|
canvasWidth = 304;
|
|
|
|
numTotalScanlines = 262;
|
|
|
|
numVisibleScanlines = 240;
|
|
|
|
defaultROMSize = 0x8000;
|
2018-11-19 14:10:13 +00:00
|
|
|
|
2019-08-05 00:10:14 +00:00
|
|
|
vdp: TMS9918A;
|
|
|
|
bios: Uint8Array;
|
|
|
|
keypadMode: boolean;
|
|
|
|
audio;
|
|
|
|
psg;
|
2018-11-19 14:10:13 +00:00
|
|
|
|
2019-08-05 00:10:14 +00:00
|
|
|
getPresets() { return ColecoVision_PRESETS; }
|
|
|
|
|
|
|
|
getKeyboardMap() { return COLECOVISION_KEYCODE_MAP; }
|
2018-11-19 14:10:13 +00:00
|
|
|
|
2019-08-05 00:10:14 +00:00
|
|
|
getVideoOptions() { return { overscan: true }; }
|
2018-11-19 14:10:13 +00:00
|
|
|
|
2019-08-05 00:10:14 +00:00
|
|
|
newRAM() {
|
|
|
|
return new Uint8Array(0x400);
|
|
|
|
}
|
|
|
|
|
|
|
|
newMembus() {
|
|
|
|
return {
|
|
|
|
read: newAddressDecoder([
|
|
|
|
[0x0000, 0x1fff, 0x1fff, (a) => { return this.bios ? this.bios[a] : 0; }],
|
|
|
|
[0x6000, 0x7fff, 0x3ff, (a) => { return this.ram[a]; }],
|
|
|
|
[0x8000, 0xffff, 0x7fff, (a) => { return this.rom ? this.rom[a] : 0; }],
|
|
|
|
]),
|
|
|
|
write: newAddressDecoder([
|
|
|
|
[0x6000, 0x7fff, 0x3ff, (a, v) => { this.ram[a] = v; }],
|
|
|
|
]),
|
|
|
|
};
|
|
|
|
}
|
2018-11-19 14:10:13 +00:00
|
|
|
|
2019-08-05 00:10:14 +00:00
|
|
|
newIOBus() {
|
|
|
|
return {
|
|
|
|
read: (addr:number):number => {
|
|
|
|
addr &= 0xff;
|
|
|
|
//console.log('IO read', hex(addr,4));
|
|
|
|
switch (addr) {
|
|
|
|
case 0xfc: return this.inputs[this.keypadMode ? 1 : 0] ^ 0xff;
|
|
|
|
case 0xff: return this.inputs[this.keypadMode ? 3 : 2] ^ 0xff;
|
|
|
|
}
|
|
|
|
if (addr >= 0xa0 && addr <= 0xbf) {
|
|
|
|
if (addr & 1)
|
|
|
|
return this.vdp.readStatus();
|
|
|
|
else
|
|
|
|
return this.vdp.readData();
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
},
|
|
|
|
write: (addr:number, val:number) => {
|
|
|
|
addr &= 0xff;
|
|
|
|
val &= 0xff;
|
|
|
|
//console.log('IO write', hex(addr,4), hex(val,2));
|
|
|
|
switch (addr >> 4) {
|
|
|
|
case 0x8: case 0x9: this.keypadMode = true; break;
|
|
|
|
case 0xc: case 0xd: this.keypadMode = false; break;
|
|
|
|
case 0xa: case 0xb:
|
2018-11-19 14:10:13 +00:00
|
|
|
if (addr & 1)
|
2019-08-05 00:10:14 +00:00
|
|
|
return this.vdp.writeAddress(val);
|
2018-11-19 14:10:13 +00:00
|
|
|
else
|
2019-08-05 00:10:14 +00:00
|
|
|
return this.vdp.writeData(val);
|
|
|
|
case 0xf: this.psg.setData(val); break;
|
2018-11-19 14:10:13 +00:00
|
|
|
}
|
2019-08-05 00:10:14 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
2018-11-19 14:10:13 +00:00
|
|
|
|
2019-08-05 00:10:14 +00:00
|
|
|
start() {
|
|
|
|
super.start();
|
|
|
|
this.bios = new lzgmini().decode(stringToByteArray(atob(COLECO_BIOS_LZG)));
|
|
|
|
this.audio = new MasterAudio();
|
|
|
|
this.psg = new SN76489_Audio(this.audio);
|
|
|
|
var cru = {
|
|
|
|
setVDPInterrupt: (b) => {
|
|
|
|
if (b) {
|
|
|
|
this.cpu.nonMaskableInterrupt();
|
|
|
|
} else {
|
|
|
|
// TODO: reset interrupt?
|
|
|
|
}
|
2018-11-19 14:10:13 +00:00
|
|
|
}
|
2019-08-05 00:10:14 +00:00
|
|
|
};
|
|
|
|
this.vdp = this.newVDP(this.video.getFrameData(), cru, true); // true = 4 sprites/line
|
|
|
|
}
|
2018-11-19 14:10:13 +00:00
|
|
|
|
2019-08-05 00:10:14 +00:00
|
|
|
newVDP(frameData, cru, flicker) {
|
|
|
|
return new TMS9918A(frameData, cru, flicker);
|
|
|
|
}
|
2018-11-19 14:10:13 +00:00
|
|
|
|
2019-08-05 00:10:14 +00:00
|
|
|
startScanline(sl: number) {
|
|
|
|
}
|
2018-11-19 14:10:13 +00:00
|
|
|
|
2019-08-05 00:10:14 +00:00
|
|
|
drawScanline(sl: number) {
|
|
|
|
this.vdp.drawScanline(sl);
|
|
|
|
}
|
2018-11-19 14:10:13 +00:00
|
|
|
|
2019-08-05 00:10:14 +00:00
|
|
|
loadState(state) {
|
|
|
|
super.loadState(state);
|
|
|
|
this.vdp.restoreState(state['vdp']);
|
|
|
|
this.keypadMode = state['kpm'];
|
|
|
|
}
|
|
|
|
saveState() {
|
|
|
|
var state = super.saveState();
|
|
|
|
state['vdp'] = this.vdp.getState();
|
|
|
|
state['kpm'] = this.keypadMode;
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
reset() {
|
|
|
|
super.reset();
|
|
|
|
this.vdp.reset();
|
|
|
|
this.psg.reset();
|
|
|
|
this.keypadMode = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
getDebugCategories() {
|
|
|
|
return super.getDebugCategories().concat(['VDP']);
|
|
|
|
}
|
|
|
|
getDebugInfo(category, state) {
|
|
|
|
switch (category) {
|
|
|
|
case 'VDP': return this.vdpStateToLongString(state.vdp);
|
|
|
|
default: return super.getDebugInfo(category, state);
|
2018-11-19 14:10:13 +00:00
|
|
|
}
|
|
|
|
}
|
2019-08-05 00:10:14 +00:00
|
|
|
vdpStateToLongString(ppu) {
|
|
|
|
return this.vdp.getRegsString();
|
|
|
|
}
|
2019-08-06 22:16:54 +00:00
|
|
|
readVRAMAddress(a : number) : number {
|
|
|
|
return this.vdp.ram[a & 0x3fff];
|
|
|
|
}
|
2018-11-19 14:10:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var COLECO_BIOS_LZG = `
|
|
|
|
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`;
|
|
|
|
|
2018-08-16 23:19:20 +00:00
|
|
|
/// MAME support
|
|
|
|
|
2018-08-26 02:00:45 +00:00
|
|
|
class ColecoVisionMAMEPlatform extends BaseMAMEPlatform implements Platform {
|
2018-08-16 23:19:20 +00:00
|
|
|
|
2018-08-23 12:49:14 +00:00
|
|
|
start() {
|
|
|
|
this.startModule(this.mainElement, {
|
2019-08-05 00:10:14 +00:00
|
|
|
jsfile: 'mamecoleco.js',
|
|
|
|
cfgfile: 'coleco.cfg',
|
|
|
|
biosfile: 'coleco/313 10031-4005 73108a.u2',
|
|
|
|
driver: 'coleco',
|
|
|
|
width: 280 * 2,
|
|
|
|
height: 216 * 2,
|
|
|
|
romfn: '/emulator/cart.rom',
|
|
|
|
romsize: 0x8000,
|
|
|
|
preInit: function(_self) {
|
2018-08-16 23:19:20 +00:00
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-08-23 12:49:14 +00:00
|
|
|
loadROM(title, data) {
|
2018-08-16 23:19:20 +00:00
|
|
|
this.loadROMFile(data);
|
|
|
|
this.loadRegion(":coleco_cart:rom", data);
|
|
|
|
}
|
|
|
|
|
2018-08-23 12:49:14 +00:00
|
|
|
getPresets() { return ColecoVision_PRESETS; }
|
2018-08-16 23:19:20 +00:00
|
|
|
|
2018-08-23 12:49:14 +00:00
|
|
|
getToolForFilename = getToolForFilename_z80;
|
|
|
|
getDefaultExtension() { return ".c"; };
|
2018-08-16 23:19:20 +00:00
|
|
|
}
|
|
|
|
|
2018-11-19 14:10:13 +00:00
|
|
|
|
2018-08-16 23:19:20 +00:00
|
|
|
///
|
|
|
|
|
2018-11-19 14:10:13 +00:00
|
|
|
PLATFORMS['coleco.mame'] = ColecoVisionMAMEPlatform;
|
2019-08-05 00:10:14 +00:00
|
|
|
PLATFORMS['coleco'] = ColecoVisionPlatform;
|