aboutsummaryrefslogtreecommitdiff
path: root/src/cpu.effekt
diff options
context:
space:
mode:
Diffstat (limited to 'src/cpu.effekt')
-rw-r--r--src/cpu.effekt60
1 files changed, 48 insertions, 12 deletions
diff --git a/src/cpu.effekt b/src/cpu.effekt
index 4a18d22..f347ddd 100644
--- a/src/cpu.effekt
+++ b/src/cpu.effekt
@@ -21,7 +21,27 @@ extern io def getNow(): Int = jsWeb "Date.now()"
interface CPU {
def initCPU(rom: ByteArray): Unit
def cycleCPU(): Unit
+
}
+def convertKey(key: String): Int = key match {
+ case "1" => 1
+ case "2" => 2
+ case "3" => 3
+ case "4" => 12
+ case "q" => 4
+ case "w" => 5
+ case "e" => 6
+ case "r" => 13
+ case "a" => 7
+ case "s" => 8
+ case "d" => 9
+ case "f" => 14
+ case "z" => 10
+ case "x" => 0
+ case "c" => 11
+ case "v" => 15
+ case _ => -1
+}
def makeCPU() {r: Renderer} = {
// Initialize the RAM
@@ -33,7 +53,6 @@ def makeCPU() {r: Renderer} = {
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
@@ -47,16 +66,27 @@ def makeCPU() {r: Renderer} = {
new CPU {
def initCPU(rom: ByteArray) = {
ram.init(rom)
- r.fill("black")
+ r.clear()
}
def cycleCPU() = {
val currentTime = getNow()
+ var key: Int = -1 // Key pressed
+ if (currentTime - last_key_update_time >= 16) { // 60Hz key updates
+ // Update keys
+ val key_ = r.getKeyPressed().getOrElse { "P" }
+ if (key_ != "P") {
+ last_key_update_time = currentTime
+ key = convertKey(key_)
+ r.log("Key pressed: " ++ show(key))
+ }
+ }
+
if (currentTime - last_instruction_run_time >= 2) { // ~500Hz
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))
+ r.log("PC: " ++ show(pc))
+ r.log("Instruction: " ++ show(inst))
// Decode
val opcode = bitwiseShr(bitwiseAnd(inst, 61440), 12)
@@ -225,16 +255,21 @@ def makeCPU() {r: Renderer} = {
pc = pc + 2
}
case 14 => {
+ r.log("nn: " ++ show(nn))
nn match {
- // Skip next instruction if key with the value of VX is pressed
+ // Skip next instruction if key with the value of VX is not pressed
case 161 => {
- // TODO: missing keyboard input
- ()
+ r.log("Key: " ++ show(key) ++ " VX: " ++ show(v.unsafeGet(x)))
+ if (key != -1 && key != v.unsafeGet(x).toInt()) {
+ pc = pc + 2
+ }
}
- // Skip next instruction if key with the value of VX is not pressed
+ // Skip next instruction if key with the value of VX is pressed
case 158 => {
- // TODO: missing keyboard input
- ()
+ r.log("Key: " ++ show(key) ++ " VX: " ++ show(v.unsafeGet(x)))
+ if (key != -1 && key == v.unsafeGet(x).toInt()) {
+ pc = pc + 2
+ }
}
} else {
r.log("Unknown Instruction: " ++ show(inst))
@@ -248,8 +283,9 @@ def makeCPU() {r: Renderer} = {
}
// Wait for a key press, store the value of the key in VX
case 10 => {
- // TODO: missing keyboard input
- ()
+ if (key != -1) {
+ v.set(x, key.toByte())
+ }
}
// SET delay timer = VX
case 21 => {