"use strict"; import { Platform, BaseMAMEPlatform, BaseZ80Platform, getToolForFilename_z80 } from "../baseplatform"; import { PLATFORMS, RAM, newAddressDecoder, padBytes, noise, setKeyboardFromMap, AnimationTimer, RasterVideo, Keys, makeKeycodeMap } from "../emu"; import { hex, lzgmini, stringToByteArray } from "../util"; import { MasterAudio, SN76489_Audio } from "../audio"; import { TMS9918A } from "../video/tms9918a"; // 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 // 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 // https://github.com/tursilion/convert9918 // http://www.harmlesslion.com/cgi-bin/showprog.cgi?ColecoVision export var ColecoVision_PRESETS = [ {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'}, ]; var COLECOVISION_KEYCODE_MAP = makeKeycodeMap([ [Keys.VK_UP, 0, 0x1], [Keys.VK_DOWN, 0, 0x4], [Keys.VK_LEFT, 0, 0x8], [Keys.VK_RIGHT, 0, 0x2], [Keys.VK_SPACE, 0, 0x40], [Keys.VK_CONTROL, 1, 0x40], [Keys.VK_W, 2, 0x1], [Keys.VK_S, 2, 0x4], [Keys.VK_A, 2, 0x8], [Keys.VK_D, 2, 0x2], [Keys.VK_Z, 2, 0x40], [Keys.VK_X, 3, 0x40], ]); /// standard emulator const _ColecoVisionPlatform = function(mainElement) { const cpuFrequency = 3579545; // MHz const canvasWidth = 304; const numTotalScanlines = 262; const numVisibleScanlines = 240; const cpuCyclesPerLine = Math.round(cpuFrequency / 60 / numTotalScanlines); var cpu, ram, membus, iobus, rom, bios; var video, vdp, timer; var audio, psg; var inputs = new Uint8Array(4); var keypadMode = false; class ColecoVisionPlatform extends BaseZ80Platform implements Platform { getPresets() { return ColecoVision_PRESETS; } start() { ram = new RAM(1024); bios = new lzgmini().decode(stringToByteArray(atob(COLECO_BIOS_LZG))); membus = { read: newAddressDecoder([ [0x0000, 0x1fff, 0x1fff, function(a) { return bios ? bios[a] : 0; }], [0x6000, 0x7fff, 0x3ff, function(a) { return ram.mem[a]; }], [0x8000, 0xffff, 0x7fff, function(a) { return rom ? rom[a] : 0; }], ]), write: newAddressDecoder([ [0x6000, 0x7fff, 0x3ff, function(a,v) { ram.mem[a] = v; }], ]), isContended: function() { return false; }, }; iobus = { read: function(addr) { addr &= 0xff; //console.log('IO read', hex(addr,4)); switch (addr) { case 0xfc: return inputs[keypadMode?1:0] ^ 0xff; case 0xff: return inputs[keypadMode?3:2] ^ 0xff; } if (addr >= 0xa0 && addr <= 0xbf) { if (addr & 1) return vdp.readStatus(); else return vdp.readData(); } return 0; }, write: function(addr, val) { addr &= 0xff; val &= 0xff; //console.log('IO write', hex(addr,4), hex(val,2)); switch (addr >> 4) { case 0x8: case 0x9: keypadMode = true; break; case 0xc: case 0xd: keypadMode = false; break; case 0xa: case 0xb: if (addr & 1) return vdp.writeAddress(val); else return vdp.writeData(val); case 0xf: psg.setData(val); break; } } }; cpu = this.newCPU(membus, iobus); video = new RasterVideo(mainElement,canvasWidth,numVisibleScanlines,{overscan:true}); video.create(); audio = new MasterAudio(); psg = new SN76489_Audio(audio); var cru = { setVDPInterrupt: (b) => { if (b) { cpu.nonMaskableInterrupt(); } else { // TODO: reset interrupt? } } }; vdp = new TMS9918A(video.getFrameData(), cru, true); // true = 4 sprites/line setKeyboardFromMap(video, inputs, COLECOVISION_KEYCODE_MAP); timer = new AnimationTimer(60, this.nextFrame.bind(this)); } readAddress(addr) { return membus.read(addr); } advance(novideo : boolean) { for (var sl=0; sl