aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/main/kotlin/cpu/Cpu.kt2
-rw-r--r--src/main/kotlin/cpu/opcodes/Rotates.kt126
2 files changed, 128 insertions, 0 deletions
diff --git a/src/main/kotlin/cpu/Cpu.kt b/src/main/kotlin/cpu/Cpu.kt
index ad006d9..fdc75b0 100644
--- a/src/main/kotlin/cpu/Cpu.kt
+++ b/src/main/kotlin/cpu/Cpu.kt
@@ -17,10 +17,12 @@ class Cpu {
stdCommandGroup.putAll(arithmetic8Bit)
stdCommandGroup.putAll(arithmetic16Bit)
stdCommandGroup.putAll(misc)
+ stdCommandGroup.putAll(rotates)
standardOpcodes = stdCommandGroup.toMap()
val extCommandGroup: MutableMap<Int, Operation> = mutableMapOf()
extCommandGroup.putAll(miscExtended)
+ extCommandGroup.putAll(rotatesExtended)
extendedOpcodes = extCommandGroup.toMap()
diff --git a/src/main/kotlin/cpu/opcodes/Rotates.kt b/src/main/kotlin/cpu/opcodes/Rotates.kt
new file mode 100644
index 0000000..1638f20
--- /dev/null
+++ b/src/main/kotlin/cpu/opcodes/Rotates.kt
@@ -0,0 +1,126 @@
+package cpu.opcodes
+
+import cpu.Operation
+import cpu.Registers
+import cpu.Registers.Flag
+import BitManipulation as bm
+
+val rotates = mapOf(
+
+ 0x07 to Operation("RLCA", 0, 4, {r, _, _ -> r.A = rotateLeft(r.A, r) }),
+
+ 0x17 to Operation("RLA", 0, 4, {r, _, _ -> r.A = rotateLeftThroughCarry(r.A, r) }),
+
+ 0x0F to Operation("RRCA", 0, 4, {r, _, _ -> r.A = rotateRight(r.A, r) }),
+
+ 0x1F to Operation("RRA", 0, 4, {r, _, _ -> r.A = rotateRightThroughCarry(r.A, r) })
+
+)
+
+val rotatesExtended = mapOf(
+
+ 0x07 to Operation("RLC A", 0, 8, {r, _, _ -> r.A = rotateLeft(r.A, r) }),
+ 0x00 to Operation("RLC B", 0, 8, {r, _, _ -> r.A = rotateLeft(r.B, r) }),
+ 0x01 to Operation("RLC C", 0, 8, {r, _, _ -> r.A = rotateLeft(r.C, r) }),
+ 0x02 to Operation("RLC D", 0, 8, {r, _, _ -> r.A = rotateLeft(r.D, r) }),
+ 0x03 to Operation("RLC E", 0, 8, {r, _, _ -> r.A = rotateLeft(r.E, r) }),
+ 0x04 to Operation("RLC H", 0, 8, {r, _, _ -> r.A = rotateLeft(r.H, r) }),
+ 0x05 to Operation("RLC L", 0, 8, {r, _, _ -> r.A = rotateLeft(r.L, r) }),
+ 0x06 to Operation("RLC (HL)", 0, 16, {r, m, _ -> m.writeByte(r.HL, rotateLeft(m.readByte(r.HL), r))}),
+
+ 0x17 to Operation("RL A", 0, 8, {r, _, _ -> r.A = rotateLeftThroughCarry(r.A, r) }),
+ 0x10 to Operation("RL B", 0, 8, {r, _, _ -> r.A = rotateLeftThroughCarry(r.B, r) }),
+ 0x11 to Operation("RL C", 0, 8, {r, _, _ -> r.A = rotateLeftThroughCarry(r.C, r) }),
+ 0x12 to Operation("RL D", 0, 8, {r, _, _ -> r.A = rotateLeftThroughCarry(r.D, r) }),
+ 0x13 to Operation("RL E", 0, 8, {r, _, _ -> r.A = rotateLeftThroughCarry(r.E, r) }),
+ 0x14 to Operation("RL H", 0, 8, {r, _, _ -> r.A = rotateLeftThroughCarry(r.H, r) }),
+ 0x15 to Operation("RL L", 0, 8, {r, _, _ -> r.A = rotateLeftThroughCarry(r.L, r) }),
+ 0x16 to Operation("RL (HL)", 0, 16, {r, m, _ -> m.writeByte(r.HL, rotateLeftThroughCarry(m.readByte(r.HL), r))}),
+
+ 0x0F to Operation("RRC A", 0, 8, {r, _, _ -> r.A = rotateRight(r.A, r) }),
+ 0x08 to Operation("RRC B", 0, 8, {r, _, _ -> r.A = rotateRight(r.B, r) }),
+ 0x09 to Operation("RRC C", 0, 8, {r, _, _ -> r.A = rotateRight(r.C, r) }),
+ 0x0A to Operation("RRC D", 0, 8, {r, _, _ -> r.A = rotateRight(r.D, r) }),
+ 0x0B to Operation("RRC E", 0, 8, {r, _, _ -> r.A = rotateRight(r.E, r) }),
+ 0x0C to Operation("RRC H", 0, 8, {r, _, _ -> r.A = rotateRight(r.H, r) }),
+ 0x0D to Operation("RRC L", 0, 8, {r, _, _ -> r.A = rotateRight(r.L, r) }),
+ 0x0E to Operation("RRC (HL)", 0, 16, {r, m, _ -> m.writeByte(r.HL, rotateRight(m.readByte(r.HL), r))}),
+
+ 0x1F to Operation("RR A", 0, 8, {r, _, _ -> r.A = rotateRightThroughCarry(r.A, r) }),
+ 0x18 to Operation("RR B", 0, 8, {r, _, _ -> r.A = rotateRightThroughCarry(r.B, r) }),
+ 0x19 to Operation("RR C", 0, 8, {r, _, _ -> r.A = rotateRightThroughCarry(r.C, r) }),
+ 0x1A to Operation("RR D", 0, 8, {r, _, _ -> r.A = rotateRightThroughCarry(r.D, r) }),
+ 0x1B to Operation("RR E", 0, 8, {r, _, _ -> r.A = rotateRightThroughCarry(r.E, r) }),
+ 0x1C to Operation("RR H", 0, 8, {r, _, _ -> r.A = rotateRightThroughCarry(r.H, r) }),
+ 0x1D to Operation("RR L", 0, 8, {r, _, _ -> r.A = rotateRightThroughCarry(r.L, r) }),
+ 0x1E to Operation("RR (HL)", 0, 16, {r, m, _ -> m.writeByte(r.HL, rotateRightThroughCarry(m.readByte(r.HL), r))})
+
+)
+
+private fun rotateLeft(n: Int, r: Registers): Int {
+
+ var result = (n shl 1) and 0xFF
+
+ r.clearFlag(Flag.SUBTRACT)
+ r.clearFlag(Flag.HALF_CARRY)
+ if (n and (1 shl 7) != 0) {
+ result = result or 1
+ r.setFlag(Flag.CARRY)
+ } else {
+ r.clearFlag(Flag.CARRY)
+ }
+
+ r.setFlagFromBool(Flag.ZERO, result == 0)
+
+ return result
+}
+
+private fun rotateLeftThroughCarry(n: Int, r: Registers): Int {
+
+ var result = (n shl 1) and 0xFF
+ result = result or r.getFlag(Flag.CARRY)
+
+ r.setFlagFromBool(Flag.ZERO, result == 0)
+ r.clearFlag(Flag.SUBTRACT)
+ r.clearFlag(Flag.HALF_CARRY)
+ r.setFlagFromBool(Flag.CARRY, (n and (1 shl 7)) != 0)
+
+ return result
+}
+
+private fun rotateRight(n: Int, r: Registers): Int {
+
+ var result = n shr 1
+
+ r.clearFlag(Flag.SUBTRACT)
+ r.clearFlag(Flag.HALF_CARRY)
+
+ if ((n and 1) == 1) {
+ result = result or (1 shl 7)
+ r.setFlag(Flag.CARRY)
+ } else {
+ r.clearFlag(Flag.CARRY)
+ }
+
+ r.setFlagFromBool(Flag.ZERO, result == 0)
+
+ return result
+}
+
+private fun rotateRightThroughCarry(n: Int, r: Registers): Int {
+
+ var result = n shr 1
+ var carry = 0
+ if (r.getFlag(Flag.CARRY) == 1){
+ carry = 1 shl 7
+ }
+
+ result = result or carry
+
+ r.setFlagFromBool(Flag.ZERO, result == 0)
+ r.clearFlag(Flag.SUBTRACT)
+ r.clearFlag(Flag.HALF_CARRY)
+ r.setFlagFromBool(Flag.CARRY, (n and 1) != 0)
+
+ return result
+} \ No newline at end of file