aboutsummaryrefslogtreecommitdiff
path: root/src/cpu.effekt
diff options
context:
space:
mode:
Diffstat (limited to 'src/cpu.effekt')
-rw-r--r--src/cpu.effekt307
1 files changed, 282 insertions, 25 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()
+ }
+ }
+}