diff options
-rw-r--r-- | src/cpu.effekt | 307 | ||||
-rw-r--r-- | src/lib.effekt | 1 | ||||
-rw-r--r-- | src/main.effekt | 15 | ||||
-rw-r--r-- | src/ram.effekt | 8 | ||||
-rw-r--r-- | src/renderer.effekt | 3 | ||||
-rw-r--r-- | src/renderers/js.effekt | 27 |
6 files changed, 318 insertions, 43 deletions
diff --git a/src/cpu.effekt b/src/cpu.effekt index 1d9fcfa..acd6c79 100644 --- a/src/cpu.effekt +++ b/src/cpu.effekt @@ -1,9 +1,11 @@ module src/cpu import bytearray +import queue import src/ram import src/renderer +import io/time /* The CPU for the Chip8 emulator. CPU Cycle and Effects: @@ -15,30 +17,285 @@ CPU Cycle and Effects: 6. Wait for next cycle */ -namespace cpu { - def run(rom: ByteArray) {r: Renderer}: Unit = { - // Initialize the RAM - var ram = makeRam() - // Load the predefined fontset into the RAM and load the ROM - ram.init(rom) - - // Initialize the CPU registers - var v: ByteArray = allocate(16) // 16 8-bit registers - var i: Int = 0 // 16-bit register treat as 12-bit !!! - - var delay: Byte = 0.toByte // Delay timer - var sound: Byte = 0.toByte // Sound timer - - var pc: Int = 512 // Program counter - var sp: Byte = 0.toByte // Stack pointer - var stack: ByteArray = allocate(32) // Stack with 16 levels. Each level is 16-bit that is 2 bytes, so 32 bytes - - // Main loop - // Fetch, Decode, Execute, Update Timers, Update Display - r.log("Starting the CPU...") - r.fill("black") - // TODO: Implement the main loop with effects - () - } +extern io def getNow(): Int = jsWeb "Date.now()" +interface CPU { + def initCPU(rom: ByteArray): Unit + def cycleCPU(): Unit } +def makeCPU() {r: Renderer} = { + // Initialize the RAM + var ram in global = makeRam() + + // Initialize the CPU registers + var v: ByteArray in global = allocate(16) // 16 8-bit registers + var i: Int in global = 0 // 16-bit register treat as 12-bit !!! + + var delay: Byte in global = 0.toByte() // Delay timer + var sound: Byte in global = 0.toByte() // Sound timer + + var pc: Int in global = 512 // Program counter + var sp: Byte in global = 0.toByte() // Stack pointer + // After checking the implementation of queue in Effekt, I realized that it also works as a stack, so I will use it as a stack + var stack in global = emptyQueue[Int](32) // Stack with 16 levels. Each level is 16-bit that is 2 bytes, so 32 bytes + + var last_key_update_time in global = getNow() + var last_instruction_run_time in global = getNow() + var last_display_update_time in global = getNow() + + new CPU { + def initCPU(rom: ByteArray) = { + ram.init(rom) + r.fill("black") + } + def cycleCPU() = { + val h = ram.getAddr(pc) + val l = ram.getAddr(pc + 1) + val inst = bitwiseOr(bitwiseShl(h.toInt(), 8), l.toInt()) + // r.log("PC: " ++ show(pc)) + // r.log("Instruction: " ++ show(inst)) + + // Decode + val opcode = bitwiseShr(bitwiseAnd(inst, 61440), 12) + val nnn = bitwiseAnd(inst, 4095) + val nn = bitwiseAnd(inst, 255) + val n = bitwiseAnd(inst, 15) + val x = bitwiseAnd(bitwiseShr(inst, 8), 15) + val y = bitwiseAnd(bitwiseShr(inst, 4), 15) + + // Execute + opcode match { + case 0 => { + nn match { + // Clear the screen + case 224 => { + r.clear() + r.log("Clearing the screen...") + pc = pc + 2 + } + // Return from a subroutine + case 238 => { + // pc = stack[Int].popFront[Int]().value[Int]() + pc = stack.popFront().getOrElse { 0 } + } + } else { + r.log("Unknown Instruction: " ++ show(inst)) + } + } + // Jump to address NNN + case 1 => { + pc = nnn + } + // Call subroutine at NNN + case 2 => { + stack.pushFront(pc + 2) + pc = nnn + } + // Skip next instruction if VX == NN + case 3 => { + if (v.unsafeGet(x).toInt() == nn) { + pc = pc + 2 + } + pc = pc + 2 + } + // Skip next instruction if VX != NN + case 4 => { + if (v.unsafeGet(x).toInt() != nn) { + pc = pc + 2 + } + pc = pc + 2 + } + // Skip next instruction if VX == VY + case 5 => { + if (v.unsafeGet(x).toInt() == v.unsafeGet(y).toInt()) { + pc = pc + 2 + } + pc = pc + 2 + } + // Set VX = NN + case 6 => { + v.set(x, nn) + pc = pc + 2 + } + // Set VX = VX + NN + case 7 => { + v.set(x, v.unsafeGet(x).toInt() + nn) + pc = pc + 2 + } + case 8 => { + n match { + // Set VX = VY + case 0 => { + v.set(x, v.unsafeGet(y)) + } + // Set VX = VX OR VY + case 1 => { + v.set(x, bitwiseOr(v.unsafeGet(x).toInt(), v.unsafeGet(y).toInt())) + } + // Set VX = VX AND VY + case 2 => { + v.set(x, bitwiseAnd(v.unsafeGet(x).toInt(), v.unsafeGet(y).toInt())) + } + // Set VX = VX XOR VY + case 3 => { + v.set(x, bitwiseXor(v.unsafeGet(x).toInt(), v.unsafeGet(y).toInt())) + } + // Set VX = VX + VY, set VF = carry + case 4 => { + val sum = v.unsafeGet(x).toInt() + v.unsafeGet(y).toInt() + v.set(15, if (sum > 255) 1 else 0) + v.set(x, sum) + + } + // Set VX = VX - VY, set VF = NOT borrow + case 5 => { + v.set(15, if (v.unsafeGet(x).toInt() > v.unsafeGet(y).toInt()) 1 else 0) + v.set(x, v.unsafeGet(x).toInt() - v.unsafeGet(y).toInt()) + } + // Set VX = VX SHR 1 + case 6 => { + v.set(15, bitwiseAnd(v.unsafeGet(x).toInt(), 1)) + v.set(x, bitwiseShr(v.unsafeGet(x).toInt(), 1)) + } + // Set VX = VY - VX, set VF = NOT borrow + case 7 => { + v.set(15, if (v.unsafeGet(y).toInt() > v.unsafeGet(x).toInt()) 1 else 0) + v.set(x, v.unsafeGet(y).toInt() - v.unsafeGet(x).toInt()) + } + // Set VX = VX SHL 1 + case 14 => { + v.set(15, bitwiseAnd(v.unsafeGet(x).toInt(), 128)) + v.set(x, bitwiseShl(v.unsafeGet(x).toInt(), 1)) + } + } else { + r.log("Unknown Instruction: " ++ show(inst)) + } + pc = pc + 2 + } + // Skip next instruction if VX != VY + case 9 => { + if (v.unsafeGet(x).toInt() != v.unsafeGet(y).toInt()) { + pc = pc + 2 + } + pc = pc + 2 + } + // Set I = NNN + case 10 => { + i = nnn + pc = pc + 2 + } + // Jump to location NNN + V0 + case 11 => { + pc = nnn + v.unsafeGet(0).toInt() + } + + // Set VX = random byte AND NN + case 12 => { + val randInt: Int = floor(random() * 255.0) + v.set(x, bitwiseAnd(randInt, nn)) + pc = pc + 2 + } + // Display n-byte sprite starting at memory location I at (VX, VY), set VF = collision + case 13 => { + val vx = v.unsafeGet(x).toInt() + val vy = v.unsafeGet(y).toInt() + var collision = false + var yline = 0 + while (yline < n) { + val pixel = ram.getAddr(i + yline) + var xline = 0 + while (xline < 8) { + if (bitwiseAnd(pixel.toInt(), bitwiseShr(128, xline)) != 0) { + if (r.get(vx + xline, vy + yline)) { + collision = true + } + r.draw(vx + xline, vy + yline) + r.log("Drawing at: " ++ show(vx + xline) ++ ", " ++ show(vy + yline)) + } + xline = xline + 1 + } + yline = yline + 1 + } + v.set(15, if (collision) 1 else 0) + pc = pc + 2 + } + case 14 => { + nn match { + // Skip next instruction if key with the value of VX is pressed + case 161 => { + // TODO: missing keyboard input + () + } + // Skip next instruction if key with the value of VX is not pressed + case 158 => { + // TODO: missing keyboard input + () + } + } else { + r.log("Unknown Instruction: " ++ show(inst)) + } + } + case 15 => { + nn match { + // Set VX = delay timer value + case 7 => { + v.set(x, delay) + } + // Wait for a key press, store the value of the key in VX + case 10 => { + // TODO: missing keyboard input + () + } + // SET delay timer = VX + case 21 => { + delay = v.unsafeGet(x) + } + // Set sound timer = VX + case 24 => { + sound = v.unsafeGet(x) + } + // SET I = I + VX + case 30 => { + i = i + v.unsafeGet(x).toInt() + } + // Set I = location of sprite for digit VX + case 41 => { + i = v.unsafeGet(x).toInt() * 5 + } + // Store BCD representation of VX in memory locations I, I+1, and I+2 + case 51 => { + val vx = v.unsafeGet(x).toInt() + ram.setAddr(i, (vx / 100).toByte()) + ram.setAddr(i + 1, mod((vx / 10), 10).toByte()) + ram.setAddr(i + 2, mod(vx, 10).toByte()) + } + // Store registers V0 through VX in memory starting at location I + case 85 => { + var index = 0 + while (index <= x) { + ram.setAddr(i + index, v.unsafeGet(index)) + index = index + 1 + } + i = i + x + 1 + } + // Read registers V0 through VX from memory starting at location I + case 101 => { + var index = 0 + while (index <= x) { + v.set(index, ram.getAddr(i + index)) + index = index + 1 + } + i = i + x + 1 + } + } else { + r.log("Unknown Instruction: " ++ show(inst)) + } + pc = pc + 2 + } + } else { + r.log("Unknown Instruction: " ++ show(inst)) + } + last_instruction_run_time = getNow() + } + } +} diff --git a/src/lib.effekt b/src/lib.effekt index d997f03..18202e1 100644 --- a/src/lib.effekt +++ b/src/lib.effekt @@ -1,3 +1,2 @@ module src/lib -def helloWorld(): String = "Hello, world!"
\ No newline at end of file diff --git a/src/main.effekt b/src/main.effekt index 627ae8e..59f8f12 100644 --- a/src/main.effekt +++ b/src/main.effekt @@ -1,16 +1,21 @@ module main // must be named same as the file! import src/cpu -import src/renderer import src/renderers/js +import src/renderer import bytearray def main(): Unit = { // Using the JS backend - def r = JSRenderer::makeRenderer - def test(rom: ByteArray) = { - cpu::run(rom) {r} + region global { + def renderer: Renderer = JSRenderer::makeRenderer + def cpu_ = makeCPU() {renderer} + def plsRender(rom: ByteArray): Unit = { + cpu_.initCPU(rom) + renderer.update(box {() => cpu_.cycleCPU()}) + } + renderer.init(plsRender) + () } - r.init(test) () }
\ No newline at end of file diff --git a/src/ram.effekt b/src/ram.effekt index 07f49cc..07a5350 100644 --- a/src/ram.effekt +++ b/src/ram.effekt @@ -3,8 +3,8 @@ module src/ram import bytearray interface Ram { - def get(address: Int): Byte - def set(address: Int, byte: Byte): Unit + def getAddr(address: Int): Byte + def setAddr(address: Int, byte: Byte): Unit def init(rom: ByteArray): Unit } @@ -46,8 +46,8 @@ def loadRom(ram: ByteArray, rom: ByteArray): Unit = rom.foreachIndex { (i, byte) def makeRam() = { var ram: ByteArray in global = allocate(4096) // 4KB of memory new Ram { - def get(address: Int) = get(ram, address) - def set(address: Int, byte: Byte) = set(ram, address, byte) + def getAddr(address: Int) = get(ram, address) + def setAddr(address: Int, byte: Byte) = set(ram, address, byte) def init(rom: ByteArray) = { loadFont(ram) loadRom(ram, rom) diff --git a/src/renderer.effekt b/src/renderer.effekt index aa8490e..358272b 100644 --- a/src/renderer.effekt +++ b/src/renderer.effekt @@ -13,6 +13,7 @@ interface Renderer { def clear(): Unit def draw(x: Int, y: Int): Unit def fill(color: String): Unit - def update(): Unit + def get(x: Int, y: Int): Bool + def update(f: () => Unit at {io, global}): Unit def log(msg: String): Unit } diff --git a/src/renderers/js.effekt b/src/renderers/js.effekt index 59b5f21..080555d 100644 --- a/src/renderers/js.effekt +++ b/src/renderers/js.effekt @@ -1,7 +1,6 @@ module src/renderers/js import src/renderer -import src/cpu import bytearray // js ffi @@ -12,7 +11,7 @@ extern io def addListener(event: String, node: Node) {handler: () => Unit}: Unit // Custom logging function that logs to the console and the page extern io def log(msg: String): Unit = jsWeb "log(${msg});" extern io def loadRom(): ByteArray = jsWeb "rom ? rom : new Uint8Array(0)" - +extern io def requestAnimationFrame {callback: () => Unit}: Unit = jsWeb "requestAnimationFrame(() => $effekt.runToplevel((ks) => ${box callback}(ks)))" extern jsWeb """ let rom = undefined; // Global Log Function @@ -20,7 +19,7 @@ extern jsWeb """ const message = new Date().toLocaleTimeString() + ' - ' + msg + ' \n' console.log(message); // Logging with timestamp - document.getElementById('logs').innerText += message; + // document.getElementById('logs').innerText += message; } function romCheck() { @@ -65,16 +64,14 @@ extern io def renderPage(content: String): Unit = jsWeb "document.write(${conten def init(run: (ByteArray) => Unit at {io, global}) = { renderPage(pageContent) val startButton = getElementById("start") - - addListener("click", startButton) {eventHandler {run}} } + def eventHandler {onClick: (ByteArray) => Unit}: Unit = { log("Start button clicked! Loading ROM...") val rom: ByteArray = loadRom() if (rom.size() != 0) { log("ROM with size: " ++ show(rom.size()) ++ " loaded!") - // TODO: Start the emulator somehow onClick(rom) } else { log("No ROM loaded! Please select a ROM file.") @@ -103,14 +100,30 @@ extern io def fill(color: String): Unit = jsWeb """ })(); """ +extern io def get(x: Int, y: Int): Bool = jsWeb """ + (() => { + const canvas = document.getElementById('canvas'); + const ctx = canvas.getContext('2d'); + const pixel = ctx.getImageData(10 * ${x}, 10 * ${y}, 1, 1).data; + return pixel[0] === 255; + })(); +""" + +extern io def update(f: () => Unit at {io, global}): Unit = jsWeb """ + setInterval(() => { + $effekt.runToplevel((ks) => ${f}(ks)); + }, 1000 / 60); +""" + namespace JSRenderer { // make JS Renderer def makeRenderer: Renderer = new Renderer { def init(run) = init(run) def clear() = clear() def draw(x: Int, y: Int) = draw(x, y) - def update() = () + def update(f: () => Unit at {io, global}) = update(f) def log(msg: String) = log(msg) def fill(color: String) = fill(color) + def get(x: Int, y: Int) = get(x, y) } } |