aboutsummaryrefslogtreecommitdiff
path: root/src/cpu.effekt
diff options
context:
space:
mode:
Diffstat (limited to 'src/cpu.effekt')
-rw-r--r--src/cpu.effekt129
1 files changed, 86 insertions, 43 deletions
diff --git a/src/cpu.effekt b/src/cpu.effekt
index f347ddd..d61d33b 100644
--- a/src/cpu.effekt
+++ b/src/cpu.effekt
@@ -71,14 +71,13 @@ def makeCPU() {r: Renderer} = {
def cycleCPU() = {
val currentTime = getNow()
var key: Int = -1 // Key pressed
- if (currentTime - last_key_update_time >= 16) { // 60Hz key updates
- // Update keys
+ if (currentTime - last_key_update_time >= 1) {
val key_ = r.getKeyPressed().getOrElse { "P" }
if (key_ != "P") {
- last_key_update_time = currentTime
key = convertKey(key_)
r.log("Key pressed: " ++ show(key))
}
+ last_key_update_time = currentTime
}
if (currentTime - last_instruction_run_time >= 2) { // ~500Hz
@@ -103,7 +102,6 @@ def makeCPU() {r: Renderer} = {
// Clear the screen
case 224 => {
r.clear()
- r.log("Clearing the screen...")
pc = pc + 2
}
// Return from a subroutine
@@ -112,7 +110,8 @@ def makeCPU() {r: Renderer} = {
pc = stack.popFront().getOrElse { 0 }
}
} else {
- r.log("Unknown Instruction: " ++ show(inst))
+ // 0nnn - SYS addr - Jump to a machine code routine at nnn
+ pc = nnn
}
}
// Jump to address NNN
@@ -130,7 +129,6 @@ def makeCPU() {r: Renderer} = {
pc = pc + 2
}
pc = pc + 2
- r.log("Skipping next instruction...")
}
// Skip next instruction if VX != NN
case 4 => {
@@ -148,59 +146,77 @@ def makeCPU() {r: Renderer} = {
}
// Set VX = NN
case 6 => {
- v.set(x, nn)
+ v.unsafeSet(x, nn.toByte())
pc = pc + 2
}
// Set VX = VX + NN
case 7 => {
- v.set(x, v.unsafeGet(x).toInt() + nn)
+ v.unsafeSet(x, (v.unsafeGet(x).toInt() + nn).toByte())
pc = pc + 2
}
case 8 => {
n match {
// Set VX = VY
case 0 => {
- v.set(x, v.unsafeGet(y))
+ v.unsafeSet(x, v.unsafeGet(y))
}
// Set VX = VX OR VY
case 1 => {
- v.set(x, bitwiseOr(v.unsafeGet(x).toInt(), v.unsafeGet(y).toInt()))
+ v.unsafeSet(x, bitwiseOr(v.unsafeGet(x).toInt(), v.unsafeGet(y).toInt()).toByte())
}
// Set VX = VX AND VY
case 2 => {
- v.set(x, bitwiseAnd(v.unsafeGet(x).toInt(), v.unsafeGet(y).toInt()))
+ v.unsafeSet(x, bitwiseAnd(v.unsafeGet(x).toInt(), v.unsafeGet(y).toInt()).toByte())
}
// Set VX = VX XOR VY
case 3 => {
- v.set(x, bitwiseXor(v.unsafeGet(x).toInt(), v.unsafeGet(y).toInt()))
+ v.unsafeSet(x, bitwiseXor(v.unsafeGet(x).toInt(), v.unsafeGet(y).toInt()).toByte())
}
// Set VX = VX + VY, set VF = carry
case 4 => {
- r.log("Adding" ++ show(v.unsafeGet(x)) ++ " and " ++ show(v.unsafeGet(y)))
val sum = v.unsafeGet(x).toInt() + v.unsafeGet(y).toInt()
- r.log("Sum: " ++ show(sum) ++ " Carry: " ++ show(if (sum > 255) 1 else 0))
- v.set(15, if (sum > 255) 1 else 0)
- v.set(x, sum.toByte())
+ v.unsafeSet(x, sum.toByte())
+ v.unsafeSet(15, (if (sum > 255) 1 else 0).toByte())
+
}
// 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())
+ val vx = v.unsafeGet(x).toInt()
+ val vy = v.unsafeGet(y).toInt()
+ val sub = vx - vy
+ r.log(show(vx) ++ "(" ++ show(x) ++ ") - " ++ show(vy) ++ "(" ++ show(y) ++ ") = " ++ show(sub) ++ "with borrow of " ++ show(if (vx > vy) 1 else 0))
+ v.unsafeSet(x, sub.toByte())
+ v.unsafeSet(15, (if (vx > vy) 1 else 0).toByte())
+ r.log("vx is now " ++ show(v.unsafeGet(x)))
+ r.log("vF is now " ++ show(v.unsafeGet(15)))
}
- // Set VX = VX SHR 1
+ // Set VX = VX SHR 1 and VF = LSB of VX
case 6 => {
- v.set(15, bitwiseAnd(v.unsafeGet(x).toInt(), 1))
- v.set(x, bitwiseShr(v.unsafeGet(x).toInt(), 1))
+ val vx = v.unsafeGet(x).toInt()
+ val lsb = bitwiseAnd(vx, 1)
+ val vx_ = bitwiseShr(vx, 1)
+ v.unsafeSet(x, vx_.toByte())
+ v.unsafeSet(15, lsb.toByte())
}
// 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())
+ val vx = v.unsafeGet(x).toInt()
+ val vy = v.unsafeGet(y).toInt()
+ val sub = vy - vx
+ r.log(show(vy) ++ "(" ++ show(y) ++ ") - " ++ show(vx) ++ "(" ++ show(x) ++ ") = " ++ show(sub) ++ "with borrow of " ++ show(if (vy > vx) 1 else 0))
+ v.unsafeSet(x, sub.toByte())
+ v.unsafeSet(15, (if (vy > vx) 1 else 0).toByte())
+ r.log("vx is now " ++ show(v.unsafeGet(x)))
+ r.log("vF is now " ++ show(v.unsafeGet(15)))
}
- // Set VX = VX SHL 1
+ // Set VX = VX SHL 1 and VF = MSB of VX
case 14 => {
- v.set(15, bitwiseAnd(v.unsafeGet(x).toInt(), 128))
- v.set(x, bitwiseShl(v.unsafeGet(x).toInt(), 1))
+ val vx = v.unsafeGet(x).toInt()
+ val msb = bitwiseAnd(vx, 128)
+ val vx_ = vx * 2
+ val vf = bitwiseShr(msb, 7)
+ v.unsafeSet(x, vx_.toByte())
+ v.unsafeSet(15, vf.toByte())
}
} else {
r.log("Unknown Instruction: " ++ show(inst))
@@ -227,7 +243,7 @@ def makeCPU() {r: Renderer} = {
// Set VX = random byte AND NN
case 12 => {
val randInt: Int = floor(random() * 255.0)
- v.set(x, bitwiseAnd(randInt, nn))
+ v.unsafeSet(x, bitwiseAnd(randInt, nn).toByte())
pc = pc + 2
}
// Display n-byte sprite starting at memory location I at (VX, VY), set VF = collision
@@ -236,38 +252,57 @@ def makeCPU() {r: Renderer} = {
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)) {
+ // Get if the current sprite pixel is set (1)
+ val spritePixel = bitwiseAnd(bitwiseShr(pixel.toInt(), 7 - xline), 1) == 1
+
+ if (spritePixel) {
+ val x = mod(vx + xline, 64)
+ val y = mod(vy + yline, 32)
+
+ // Get current screen pixel
+ val screenPixel = r.get(x, y)
+
+ // XOR operation
+ if (screenPixel == spritePixel) {
+ // If both pixels are on, turn it off and set collision
+ r.draw(x, y, "black")
collision = true
+ } else if (screenPixel != spritePixel) {
+ // If screen pixel is off and sprite pixel is on, turn it on
+ r.draw(x, y, "white")
}
- 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)
+
+ v.unsafeSet(15, (if (collision) 1 else 0).toByte())
pc = pc + 2
}
case 14 => {
- r.log("nn: " ++ show(nn))
nn match {
// Skip next instruction if key with the value of VX is not pressed
case 161 => {
- r.log("Key: " ++ show(key) ++ " VX: " ++ show(v.unsafeGet(x)))
- if (key != -1 && key != v.unsafeGet(x).toInt()) {
+ val vx = v.unsafeGet(x).toInt()
+ if (key != v.unsafeGet(x).toInt()) {
+ pc = pc + 4
+ } else {
pc = pc + 2
}
}
// Skip next instruction if key with the value of VX is pressed
case 158 => {
- r.log("Key: " ++ show(key) ++ " VX: " ++ show(v.unsafeGet(x)))
+ val vx = v.unsafeGet(x).toInt()
if (key != -1 && key == v.unsafeGet(x).toInt()) {
+ pc = pc + 4
+ } else {
pc = pc + 2
}
}
@@ -279,29 +314,35 @@ def makeCPU() {r: Renderer} = {
nn match {
// Set VX = delay timer value
case 7 => {
- v.set(x, delay)
+ v.unsafeSet(x, delay)
+ pc = pc + 2
}
// Wait for a key press, store the value of the key in VX
case 10 => {
- if (key != -1) {
- v.set(x, key.toByte())
+ if (key >= 0) {
+ v.unsafeSet(x, key.toByte())
+ pc = pc + 2
}
}
// SET delay timer = VX
case 21 => {
delay = v.unsafeGet(x)
+ pc = pc + 2
}
// Set sound timer = VX
case 24 => {
sound = v.unsafeGet(x)
+ pc = pc + 2
}
// SET I = I + VX
case 30 => {
i = i + v.unsafeGet(x).toInt()
+ pc = pc + 2
}
// Set I = location of sprite for digit VX
case 41 => {
i = v.unsafeGet(x).toInt() * 5
+ pc = pc + 2
}
// Store BCD representation of VX in memory locations I, I+1, and I+2
case 51 => {
@@ -309,6 +350,7 @@ def makeCPU() {r: Renderer} = {
ram.setAddr(i, (vx / 100).toByte())
ram.setAddr(i + 1, mod((vx / 10), 10).toByte())
ram.setAddr(i + 2, mod(vx, 10).toByte())
+ pc = pc + 2
}
// Store registers V0 through VX in memory starting at location I
case 85 => {
@@ -317,21 +359,22 @@ def makeCPU() {r: Renderer} = {
ram.setAddr(i + index, v.unsafeGet(index))
index = index + 1
}
- i = i + x + 1
+ // i = i + x + 1
+ pc = pc + 2
}
// 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))
+ v.unsafeSet(index, ram.getAddr(i + index))
index = index + 1
}
- i = i + x + 1
+ // i = i + x + 1
+ pc = pc + 2
}
} else {
r.log("Unknown Instruction: " ++ show(inst))
}
- pc = pc + 2
}
} else {
r.log("Unknown Instruction: " ++ show(inst))