Skip to content

Commit 66cf6c4

Browse files
committedJan 20, 2024
cpu: full cycle-accurate cpu emulation
Signed-off-by: Sebastian Jastrzebski <sebby2k@gmail.com>
1 parent 09cc8f4 commit 66cf6c4

21 files changed

+2590
-140
lines changed
 

‎README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -227,7 +227,7 @@ I've included a number of examples from Kick Assembler that I've used to test va
227227

228228
The cpu validation was performed with the help of [Klaus2m5 functional tests](https://github.com/Klaus2m5/6502_65C02_functional_tests) for the 6502 processor
229229

230-
./target/release/zinc64 --binary bin/6502_functional_test.bin --offset=1024 --console --loglevel trace
230+
./target/release/zinc64 --console --loglevel trace bin/6502_functional_test.bin
231231

232232
## Keyboard Shortcuts
233233

‎TODO ‎TODO.md

+1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
# TODO
22

3+
- http://oxyron.de/html/opcodes02.html
34
- fix cpu usage
45
- app: add screenshot support
56
- app: add ui/menus

‎logger.conf

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
app=error
2-
cpu::ins=trace
2+
cpu::ins=error
33
cpu::int=trace
4+
cpu::uop=error
45
mem::banks=trace
56
cart::banks=trace
67
cia::reg=error

‎zinc64-core/src/cpu/cpu.rs

+1,075
Large diffs are not rendered by default.

‎zinc64-core/src/cpu/mod.rs

+3-6
Original file line numberDiff line numberDiff line change
@@ -2,10 +2,7 @@
22
// Copyright (c) 2016-2019 Sebastian Jastrzebski. All rights reserved.
33
// Licensed under the GPLv3. See LICENSE file in the project root for full license text.
44

5-
mod cpu6510;
6-
mod instruction;
7-
mod operand;
5+
mod cpu;
6+
mod uops;
87

9-
pub use self::cpu6510::Cpu6510;
10-
pub use self::instruction::Instruction;
11-
pub use self::operand::Operand;
8+
pub use self::cpu::Cpu6510;

‎zinc64-core/src/cpu/uops.rs

+818
Large diffs are not rendered by default.

‎zinc64-core/src/cpu/cpu6510.rs ‎zinc64-core/src/cpu_gen1/cpu6510.rs

+65-58
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44

55
#![cfg_attr(feature = "cargo-clippy", allow(clippy::cast_lossless))]
66

7-
use crate::factory::{make_noop, Addressable, Cpu, TickFn};
7+
use crate::factory::{make_noop, Addressable, Cpu, Register, TickFn};
88
use crate::util::{IoPort, IrqLine, Pin, Shared};
99
use core::fmt;
1010
use log::LogLevel;
@@ -51,13 +51,13 @@ impl Interrupt {
5151
}
5252
}
5353

54-
struct Registers {
55-
a: u8,
56-
x: u8,
57-
y: u8,
58-
sp: u8,
59-
pc: u16,
60-
p: u8,
54+
pub struct Registers {
55+
pub a: u8,
56+
pub x: u8,
57+
pub y: u8,
58+
pub sp: u8,
59+
pub pc: u16,
60+
pub p: u8,
6161
}
6262

6363
impl Registers {
@@ -86,8 +86,9 @@ pub struct Cpu6510 {
8686
// Dependencies
8787
mem: Shared<dyn Addressable>,
8888
// Runtime State
89-
regs: Registers,
89+
pub regs: Registers,
9090
last_nmi: bool,
91+
last_pc: u16,
9192
// I/O
9293
ba_line: Shared<Pin>,
9394
io_port: Shared<IoPort>,
@@ -107,6 +108,7 @@ impl Cpu6510 {
107108
mem,
108109
regs: Registers::new(),
109110
last_nmi: false,
111+
last_pc: 0,
110112
ba_line,
111113
io_port,
112114
irq_line,
@@ -622,66 +624,59 @@ impl Cpu6510 {
622624
}
623625

624626
impl Cpu for Cpu6510 {
625-
fn get_a(&self) -> u8 {
626-
self.regs.a
627+
628+
fn get_register(&self, reg: Register) -> u8 {
629+
match reg {
630+
Register::A => self.regs.a,
631+
Register::X => self.regs.x,
632+
Register::Y => self.regs.y,
633+
Register::SP => self.regs.sp,
634+
Register::PCL => self.regs.pc as u8,
635+
Register::PCH => (self.regs.pc >> 8) as u8,
636+
Register::P => self.regs.p,
637+
}
627638
}
628639

629-
fn get_p(&self) -> u8 {
630-
self.regs.p
640+
fn set_register(&mut self, reg: Register, value: u8) {
641+
match reg {
642+
Register::A => {
643+
self.regs.a = value;
644+
}
645+
Register::X => {
646+
self.regs.x = value;
647+
}
648+
Register::Y => {
649+
self.regs.y = value;
650+
}
651+
Register::SP => {
652+
self.regs.sp = value;
653+
}
654+
Register::PCL => {
655+
self.regs.pc = (self.regs.pc & 0xff00) | u16::from(value);
656+
}
657+
Register::PCH => {
658+
self.regs.pc = (u16::from(value) << 8) | (self.regs.pc & 0xff);
659+
}
660+
Register::P => {
661+
self.regs.p = value;
662+
}
663+
}
631664
}
632665

633666
fn get_pc(&self) -> u16 {
634667
self.regs.pc
635668
}
636669

637-
fn get_sp(&self) -> u8 {
638-
self.regs.sp
639-
}
640-
641-
fn get_x(&self) -> u8 {
642-
self.regs.x
643-
}
644-
645-
fn get_y(&self) -> u8 {
646-
self.regs.y
647-
}
648-
649-
fn set_a(&mut self, value: u8) {
650-
self.regs.a = value;
651-
}
652-
653-
fn set_p(&mut self, value: u8) {
654-
self.regs.p = value;
655-
}
656-
657670
fn set_pc(&mut self, value: u16) {
658671
self.regs.pc = value;
659672
}
660673

661-
fn set_sp(&mut self, value: u8) {
662-
self.regs.sp = value;
663-
}
664-
665-
fn set_x(&mut self, value: u8) {
666-
self.regs.x = value;
667-
}
668-
669-
fn set_y(&mut self, value: u8) {
670-
self.regs.y = value;
671-
}
672-
673-
fn reset(&mut self) {
674-
self.regs.reset();
675-
self.last_nmi = false;
676-
self.io_port.borrow_mut().set_value(0xff);
677-
self.irq_line.borrow_mut().reset();
678-
self.nmi_line.borrow_mut().reset();
679-
self.write(0x0000, 0b_0010_1111);
680-
self.write(0x0001, 0b_0001_1111);
681-
self.interrupt(&Interrupt::Reset, &make_noop());
674+
fn is_cpu_jam(&self) -> bool {
675+
self.last_pc == self.get_pc()
682676
}
683677

684678
fn step(&mut self, tick_fn: &TickFn) {
679+
self.last_pc = self.get_pc();
685680
while self.ba_line.borrow().is_low() {
686681
tick_fn();
687682
}
@@ -701,6 +696,18 @@ impl Cpu for Cpu6510 {
701696
self.last_nmi = self.nmi_line.borrow().is_low();
702697
}
703698

699+
fn reset(&mut self) {
700+
self.regs.reset();
701+
self.last_nmi = false;
702+
self.last_pc = 0;
703+
self.io_port.borrow_mut().set_value(0xff);
704+
self.irq_line.borrow_mut().reset();
705+
self.nmi_line.borrow_mut().reset();
706+
self.write(0x0000, 0b_0010_1111);
707+
self.write(0x0001, 0b_0001_1111);
708+
self.interrupt(&Interrupt::Reset, &make_noop());
709+
}
710+
704711
// -- I/O
705712

706713
fn read(&self, address: u16) -> u8 {
@@ -798,10 +805,10 @@ mod tests {
798805
#[test]
799806
fn adc_80_16() {
800807
let mut cpu = setup_cpu();
801-
cpu.set_a(80);
808+
cpu.set_register(Register::A, 80);
802809
cpu.set_flag(Flag::Carry, false);
803810
cpu.execute(&Instruction::ADC(Operand::Immediate(16)), &make_noop());
804-
assert_eq!(96, cpu.get_a());
811+
assert_eq!(96, cpu.get_register(Register::A));
805812
assert_eq!(false, cpu.test_flag(Flag::Carry));
806813
assert_eq!(false, cpu.test_flag(Flag::Negative));
807814
assert_eq!(false, cpu.test_flag(Flag::Overflow));
@@ -810,9 +817,9 @@ mod tests {
810817
#[test]
811818
fn inc_with_overflow() {
812819
let mut cpu = setup_cpu();
813-
cpu.set_a(0xff);
820+
cpu.set_register(Register::A, 0xff);
814821
cpu.execute(&Instruction::INC(Operand::Accumulator), &make_noop());
815-
assert_eq!(0x00, cpu.get_a());
822+
assert_eq!(0x00, cpu.get_register(Register::A));
816823
assert_eq!(false, cpu.test_flag(Flag::Negative));
817824
assert_eq!(true, cpu.test_flag(Flag::Zero));
818825
}
File renamed without changes.

‎zinc64-core/src/cpu_gen1/mod.rs

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
// This file is part of zinc64.
2+
// Copyright (c) 2016-2019 Sebastian Jastrzebski. All rights reserved.
3+
// Licensed under the GPLv3. See LICENSE file in the project root for full license text.
4+
5+
mod cpu6510;
6+
mod instruction;
7+
mod operand;
8+
9+
pub use self::cpu6510::Cpu6510;
10+
pub use self::instruction::Instruction;
11+
pub use self::operand::Operand;

‎zinc64-core/src/cpu/operand.rs ‎zinc64-core/src/cpu_gen1/operand.rs

+22-22
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44

55
#![cfg_attr(feature = "cargo-clippy", allow(clippy::cast_lossless))]
66

7-
use crate::factory::{Cpu, TickFn};
7+
use crate::factory::{Cpu, TickFn, Register};
88
use core::fmt;
99

1010
use super::Cpu6510;
@@ -41,27 +41,27 @@ impl Operand {
4141
// FIXME cpu: rmw
4242
tick_fn();
4343
}
44-
address.wrapping_add(cpu.get_x()) as u16
44+
address.wrapping_add(cpu.regs.x) as u16
4545
}
4646
Operand::ZeroPageY(address) => {
4747
tick_fn();
48-
address.wrapping_add(cpu.get_y()) as u16
48+
address.wrapping_add(cpu.regs.y) as u16
4949
}
5050
Operand::Absolute(address) => address,
5151
Operand::AbsoluteX(address) => {
5252
if rmw {
5353
tick_fn();
5454
}
55-
address.wrapping_add(cpu.get_x() as u16)
55+
address.wrapping_add(cpu.regs.x as u16)
5656
}
5757
Operand::AbsoluteY(address) => {
5858
if rmw {
5959
tick_fn();
6060
}
61-
address.wrapping_add(cpu.get_y() as u16)
61+
address.wrapping_add(cpu.regs.y as u16)
6262
}
6363
Operand::IndirectX(address) => {
64-
let calc_address = address.wrapping_add(cpu.get_x()) as u16;
64+
let calc_address = address.wrapping_add(cpu.regs.x) as u16;
6565
tick_fn();
6666
cpu.read_internal_u16(calc_address, tick_fn)
6767
}
@@ -70,7 +70,7 @@ impl Operand {
7070
tick_fn();
7171
}
7272
cpu.read_internal_u16(address as u16, tick_fn)
73-
.wrapping_add(cpu.get_y() as u16)
73+
.wrapping_add(cpu.regs.y as u16)
7474
}
7575
Operand::Indirect(address) => cpu.read_internal_u16(address, tick_fn),
7676
Operand::Relative(offset) => {
@@ -89,7 +89,7 @@ impl Operand {
8989

9090
pub fn get(&self, cpu: &Cpu6510, tick_fn: &TickFn) -> u8 {
9191
match *self {
92-
Operand::Accumulator => cpu.get_a(),
92+
Operand::Accumulator => cpu.get_register(Register::A),
9393
Operand::Immediate(value) => value,
9494
Operand::Indirect(_) => panic!("illegal op for addressing mode {}", "indirect"),
9595
Operand::Relative(_) => panic!("illegal op for addressing mode {}", "relative"),
@@ -102,7 +102,7 @@ impl Operand {
102102

103103
pub fn set(&self, cpu: &mut Cpu6510, value: u8, rmw: bool, tick_fn: &TickFn) {
104104
match *self {
105-
Operand::Accumulator => cpu.set_a(value),
105+
Operand::Accumulator => cpu.set_register(Register::A, value),
106106
Operand::Immediate(_) => panic!("illegal op for addressing mode {}", "immediate"),
107107
Operand::Indirect(_) => panic!("illegal op for addressing mode {}", "indirect"),
108108
Operand::Relative(_) => panic!("illegal op for addressing mode {}", "relative"),
@@ -178,31 +178,31 @@ mod tests {
178178
#[test]
179179
fn ea_zeropage_x() {
180180
let mut cpu = setup_cpu();
181-
cpu.set_x(0x01);
181+
cpu.set_register(Register::X, 0x01);
182182
let op = Operand::ZeroPageX(0x10);
183183
assert_eq!(0x0011, op.ea(&cpu, false, &make_noop()));
184184
}
185185

186186
#[test]
187187
fn ea_zeropage_x_wrapping() {
188188
let mut cpu = setup_cpu();
189-
cpu.set_x(0x03);
189+
cpu.set_register(Register::X, 0x03);
190190
let op = Operand::ZeroPageX(0xff);
191191
assert_eq!(0x0002, op.ea(&cpu, false, &make_noop()));
192192
}
193193

194194
#[test]
195195
fn ea_zeropage_y() {
196196
let mut cpu = setup_cpu();
197-
cpu.set_y(0x01);
197+
cpu.set_register(Register::Y, 0x01);
198198
let op = Operand::ZeroPageY(0x10);
199199
assert_eq!(0x0011, op.ea(&cpu, false, &make_noop()));
200200
}
201201

202202
#[test]
203203
fn ea_zeropage_y_wrapping() {
204204
let mut cpu = setup_cpu();
205-
cpu.set_y(0x03);
205+
cpu.set_register(Register::Y, 0x03);
206206
let op = Operand::ZeroPageY(0xff);
207207
assert_eq!(0x0002, op.ea(&cpu, false, &make_noop()));
208208
}
@@ -217,31 +217,31 @@ mod tests {
217217
#[test]
218218
fn ea_absolute_x() {
219219
let mut cpu = setup_cpu();
220-
cpu.set_x(0x01);
220+
cpu.set_register(Register::X, 0x01);
221221
let op = Operand::AbsoluteX(0x0100);
222222
assert_eq!(0x0101, op.ea(&cpu, false, &make_noop()));
223223
}
224224

225225
#[test]
226226
fn ea_absolute_x_wrapping() {
227227
let mut cpu = setup_cpu();
228-
cpu.set_x(0x03);
228+
cpu.set_register(Register::X, 0x03);
229229
let op = Operand::AbsoluteX(0xffff);
230230
assert_eq!(0x0002, op.ea(&cpu, false, &make_noop()));
231231
}
232232

233233
#[test]
234234
fn ea_absolute_y() {
235235
let mut cpu = setup_cpu();
236-
cpu.set_y(0x01);
236+
cpu.set_register(Register::Y, 0x01);
237237
let op = Operand::AbsoluteY(0x0100);
238238
assert_eq!(0x0101, op.ea(&cpu, false, &make_noop()));
239239
}
240240

241241
#[test]
242242
fn ea_absolute_y_wrapping() {
243243
let mut cpu = setup_cpu();
244-
cpu.set_y(0x03);
244+
cpu.set_register(Register::Y, 0x03);
245245
let op = Operand::AbsoluteY(0xffff);
246246
assert_eq!(0x0002, op.ea(&cpu, false, &make_noop()));
247247
}
@@ -251,7 +251,7 @@ mod tests {
251251
let mut cpu = setup_cpu();
252252
cpu.write_internal(0x0006, 0x00, &make_noop());
253253
cpu.write_internal(0x0007, 0x16, &make_noop());
254-
cpu.set_x(0x05);
254+
cpu.set_register(Register::X, 0x05);
255255
let op = Operand::IndirectX(0x01);
256256
assert_eq!(0x1600, op.ea(&cpu, false, &make_noop()));
257257
}
@@ -261,7 +261,7 @@ mod tests {
261261
let mut cpu = setup_cpu();
262262
cpu.write_internal(0x0006, 0x00, &make_noop());
263263
cpu.write_internal(0x0007, 0x16, &make_noop());
264-
cpu.set_x(0x07);
264+
cpu.set_register(Register::X, 0x07);
265265
let op = Operand::IndirectX(0xff);
266266
assert_eq!(0x1600, op.ea(&cpu, false, &make_noop()));
267267
}
@@ -271,7 +271,7 @@ mod tests {
271271
let mut cpu = setup_cpu();
272272
cpu.write_internal(0x0006, 0x00, &make_noop());
273273
cpu.write_internal(0x0007, 0x16, &make_noop());
274-
cpu.set_y(0x05);
274+
cpu.set_register(Register::Y, 0x05);
275275
let op = Operand::IndirectY(0x06);
276276
assert_eq!(0x1605, op.ea(&cpu, false, &make_noop()));
277277
}
@@ -281,7 +281,7 @@ mod tests {
281281
let mut cpu = setup_cpu();
282282
cpu.write_internal(0x0006, 0xff, &make_noop());
283283
cpu.write_internal(0x0007, 0xff, &make_noop());
284-
cpu.set_y(0x06);
284+
cpu.set_register(Register::Y, 0x06);
285285
let op = Operand::IndirectY(0x06);
286286
assert_eq!(0x0005, op.ea(&cpu, false, &make_noop()));
287287
}
@@ -313,7 +313,7 @@ mod tests {
313313
#[test]
314314
fn get_accumulator() {
315315
let mut cpu = setup_cpu();
316-
cpu.set_a(0xab);
316+
cpu.set_register(Register::A, 0xab);
317317
let op = Operand::Accumulator;
318318
assert_eq!(0xab, op.get(&cpu, &make_noop()));
319319
}

‎zinc64-core/src/factory/types.rs

+17-11
Original file line numberDiff line numberDiff line change
@@ -61,24 +61,30 @@ pub trait Chip {
6161
fn write(&mut self, reg: u8, value: u8);
6262
}
6363

64+
#[derive(Copy, Clone)]
65+
pub enum Register {
66+
A,
67+
X,
68+
Y,
69+
SP,
70+
PCL,
71+
PCH,
72+
P
73+
}
74+
6475
/// CPU is responsible for decoding and executing instructions.
6576
pub trait Cpu {
66-
fn get_a(&self) -> u8;
67-
fn get_p(&self) -> u8;
77+
// -- Getters/Setters
78+
fn get_register(&self, reg: Register) -> u8;
79+
fn set_register(&mut self, reg: Register, value: u8);
6880
fn get_pc(&self) -> u16;
69-
fn get_sp(&self) -> u8;
70-
fn get_x(&self) -> u8;
71-
fn get_y(&self) -> u8;
72-
fn set_a(&mut self, value: u8);
73-
fn set_p(&mut self, value: u8);
7481
fn set_pc(&mut self, value: u16);
75-
fn set_sp(&mut self, value: u8);
76-
fn set_x(&mut self, value: u8);
77-
fn set_y(&mut self, value: u8);
78-
fn reset(&mut self);
82+
fn is_cpu_jam(&self) -> bool;
7983
/// The core method of the cpu, decodes and executes one instruction. Tick callback is invoked
8084
/// for each elapsed clock cycle.
8185
fn step(&mut self, tick_fn: &TickFn);
86+
/// Reset chip.
87+
fn reset(&mut self);
8288
// I/O
8389
/// Read byte from the specified address.
8490
fn read(&self, address: u16) -> u8;

‎zinc64-core/src/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ extern crate core;
1313
extern crate log;
1414

1515
pub mod cpu;
16+
pub mod cpu_gen1;
1617
pub mod device;
1718
pub mod factory;
1819
pub mod io;

‎zinc64-core/tests/cpu_gen1_timing.rs

+326
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,326 @@
1+
// This file is part of zinc64.
2+
// Copyright (c) 2016-2019 Sebastian Jastrzebski. All rights reserved.
3+
// Licensed under the GPLv3. See LICENSE file in the project root for full license text.
4+
5+
use std::cell::{Cell, RefCell};
6+
use std::rc::Rc;
7+
8+
use zinc64_core::cpu_gen1::Cpu6510;
9+
use zinc64_core::factory::{Addressable, Cpu, TickFn};
10+
use zinc64_core::util::{IoPort, IrqLine, Pin, Ram};
11+
12+
struct MockMemory {
13+
ram: Ram,
14+
}
15+
16+
impl MockMemory {
17+
pub fn new(ram: Ram) -> Self {
18+
MockMemory { ram }
19+
}
20+
}
21+
22+
impl Addressable for MockMemory {
23+
fn read(&self, address: u16) -> u8 {
24+
self.ram.read(address)
25+
}
26+
27+
fn write(&mut self, address: u16, value: u8) {
28+
self.ram.write(address, value);
29+
}
30+
}
31+
32+
fn setup_cpu() -> Cpu6510 {
33+
let ba_line = Rc::new(RefCell::new(Pin::new_high()));
34+
let cpu_io_port = Rc::new(RefCell::new(IoPort::new(0x00, 0xff)));
35+
let cpu_irq = Rc::new(RefCell::new(IrqLine::new("irq")));
36+
let cpu_nmi = Rc::new(RefCell::new(IrqLine::new("nmi")));
37+
let mem = Rc::new(RefCell::new(MockMemory::new(Ram::new(0x10000))));
38+
Cpu6510::new(mem, cpu_io_port, ba_line, cpu_irq, cpu_nmi)
39+
}
40+
41+
// Based on 65xx Processor Data from http://www.romhacking.net/documents/318/
42+
43+
const OPCODE_TIMING: [u8; 256] = [
44+
7, // 00 BRK #$ab
45+
6, // 01 ORA ($ab,X)
46+
0, // 02 HLT*
47+
0, // 03 ASO* ($ab,X)
48+
0, // 04 SKB* $ab
49+
3, // 05 ORA $ab
50+
5, // 06 ASL $ab
51+
0, // 07 ASO* $ab
52+
3, // 08 PHP
53+
2, // 09 ORA #$ab
54+
2, // 0A ASL A
55+
0, // 0B ANC* #$ab
56+
0, // 0C SKW* $abcd
57+
4, // 0D ORA $abcd
58+
6, // 0E ASL $abcd
59+
0, // 0F ASO* $abcd
60+
2, // 10 BPL nearlabel
61+
5, // 11 ORA ($ab),Y
62+
0, // 12 HLT*
63+
0, // 13 ASO* ($ab),Y
64+
0, // 14 SKB* $ab,X
65+
4, // 15 ORA $ab,X
66+
6, // 16 ASL $ab,X
67+
0, // 17 ASO* $ab,X
68+
2, // 18 CLC
69+
4, // 19 ORA $abcd,Y
70+
0, // 1A NOP*
71+
0, // 1B ASO* $abcd,Y
72+
0, // 1C SKW* $abcd,X
73+
4, // 1D ORA $abcd,X
74+
7, // 1E ASL $abcd,X
75+
0, // 1F ASO* $abcd,X
76+
6, // 20 JSR $abcd
77+
6, // 21 AND ($ab,X)
78+
0, // 22 HLT*
79+
0, // 23 RLA* ($ab,X)
80+
3, // 24 BIT $ab
81+
3, // 25 AND $ab
82+
5, // 26 ROL $ab
83+
0, // 27 RLA* $ab
84+
4, // 28 PLP
85+
2, // 29 AND #$ab
86+
2, // 2A ROL A
87+
0, // 2B ANC* #$ab
88+
4, // 2C BIT $abcd
89+
4, // 2D AND $abcd
90+
6, // 2E ROL $abcd
91+
0, // 2F RLA* $abcd
92+
2, // 30 BMI nearlabel
93+
5, // 31 AND ($ab),Y
94+
0, // 32 HLT*
95+
0, // 33 RLA* ($ab),Y
96+
0, // 34 SKB* $ab,X
97+
4, // 35 AND $ab,X
98+
6, // 36 ROL $ab,X
99+
0, // 37 RLA* $ab,X
100+
2, // 38 SEC
101+
4, // 39 AND $abcd,Y
102+
0, // 3A NOP*
103+
0, // 3B RLA* $abcd,Y
104+
0, // 3C SKW* $abcd,X
105+
4, // 3D AND $abcd,X
106+
7, // 3E ROL $abcd,X
107+
0, // 3F RLA* $abcd,X
108+
6, // 40 RTI
109+
6, // 41 EOR ($ab,X)
110+
0, // 42 HLT*
111+
8, // 43 LSE* ($ab,X)
112+
0, // 44 SKB* $ab
113+
3, // 45 EOR $ab
114+
5, // 46 LSR $ab
115+
5, // 47 LSE* $ab
116+
3, // 48 PHA
117+
2, // 49 EOR #$ab
118+
2, // 4A LSR A
119+
2, // 4B ALR* #$ab
120+
3, // 4C JMP $abcd
121+
4, // 4D EOR $abcd
122+
6, // 4E LSR $abcd
123+
6, // 4F LSE* $abcd
124+
2, // 50 BVC nearlabel
125+
5, // 51 EOR ($ab),Y
126+
0, // 52 HLT*
127+
8, // 53 LSE* ($ab),Y
128+
0, // 54 SKB* $ab,X
129+
4, // 55 EOR $ab,X
130+
6, // 56 LSR $ab,X
131+
6, // 57 LSE* $ab,X
132+
2, // 58 CLI
133+
4, // 59 EOR $abcd,Y
134+
0, // 5A NOP*
135+
7, // 5B LSE* $abcd,Y
136+
0, // 5C SKW* $abcd,X
137+
4, // 5D EOR $abcd,X
138+
7, // 5E LSR $abcd,X
139+
7, // 5F LSE* $abcd,X
140+
6, // 60 RTS
141+
6, // 61 ADC ($ab,X)
142+
0, // 62 HLT*
143+
0, // 63 RRA* ($ab,X)
144+
0, // 64 SKB* $ab
145+
3, // 65 ADC $ab
146+
5, // 66 ROR $ab
147+
0, // 67 RRA* $ab
148+
4, // 68 PLA
149+
2, // 69 ADC #$ab
150+
2, // 6A ROR A
151+
0, // 6B ARR* #$ab
152+
5, // 6C JMP ($abcd)
153+
4, // 6D ADC $abcd
154+
6, // 6E ROR $abcd
155+
0, // 6F RRA* $abcd
156+
2, // 70 BVS nearlabel
157+
5, // 71 ADC ($ab),Y
158+
0, // 72 HLT*
159+
0, // 73 RRA* ($ab),Y
160+
0, // 74 SKB* $ab,X
161+
4, // 75 ADC $ab,X
162+
6, // 76 ROR $ab,X
163+
0, // 77 RRA* $ab,X
164+
2, // 78 SEI
165+
4, // 79 ADC $abcd,Y
166+
0, // 7A NOP*
167+
0, // 7B RRA* $abcd,Y
168+
0, // 7C SKW* $abcd,X
169+
4, // 7D ADC $abcd,X
170+
7, // 7E ROR $abcd,X
171+
0, // 7F RRA* $abcd,X
172+
3, // FIXME 80 SKB* #$ab
173+
6, // 81 STA ($ab,X)
174+
0, // 82 SKB* #$ab
175+
0, // 83 SAX* ($ab,X)
176+
3, // 84 STY $ab
177+
3, // 85 STA $ab
178+
3, // 86 STX $ab
179+
0, // 87 SAX* $ab
180+
2, // 88 DEY
181+
0, // 89 SKB* #$ab
182+
2, // 8A TXA
183+
2, // 8B ANE* #$ab
184+
4, // 8C STY $abcd
185+
4, // 8D STA $abcd
186+
4, // 8E STX $abcd
187+
0, // 8F SAX* $abcd
188+
2, // 90 BCC nearlabel
189+
6, // 91 STA ($ab),Y
190+
0, // 92 HLT*
191+
0, // 93 SHA* ($ab),Y
192+
4, // 94 STY $ab,X
193+
4, // 95 STA $ab,X
194+
4, // 96 STX $ab,Y
195+
0, // 97 SAX* $ab,Y
196+
2, // 98 TYA
197+
5, // 99 STA $abcd,Y
198+
2, // 9A TXS
199+
0, // 9B SHS* $abcd,Y
200+
0, // 9C SHY* $abcd,X
201+
5, // 9D STA $abcd,X
202+
0, // 9E SHX* $abcd,Y
203+
0, // 9F SHA* $abcd,Y
204+
2, // A0 LDY #$ab
205+
6, // A1 LDA ($ab,X)
206+
2, // A2 LDX #$ab
207+
6, // A3 LAX* ($ab,X)
208+
3, // A4 LDY $ab
209+
3, // A5 LDA $ab
210+
3, // A6 LDX $ab
211+
3, // A7 LAX* $ab
212+
2, // A8 TAY
213+
2, // A9 LDA #$ab
214+
2, // AA TAX
215+
2, // AB ANX* #$ab
216+
4, // AC LDY $abcd
217+
4, // AD LDA $abcd
218+
4, // AE LDX $abcd
219+
4, // AF LAX* $abcd
220+
2, // B0 BCS nearlabel
221+
5, // B1 LDA ($ab),Y
222+
0, // B2 HLT*
223+
5, // B3 LAX* ($ab),Y
224+
4, // B4 LDY $ab,X
225+
4, // B5 LDA $ab,X
226+
4, // B6 LDX $ab,Y
227+
4, // B7 LAX* $ab,Y
228+
2, // B8 CLV
229+
4, // B9 LDA $abcd,Y
230+
2, // BA TSX
231+
0, // BB LAS* $abcd,Y
232+
4, // BC LDY $abcd,X
233+
4, // BD LDA $abcd,X
234+
4, // BE LDX $abcd,Y
235+
4, // BF LAX* $abcd,Y
236+
2, // C0 CPY #$ab
237+
6, // C1 CMP ($ab,X)
238+
0, // C2 SKB* #$ab
239+
0, // C3 DCM* ($ab,X)
240+
3, // C4 CPY $ab
241+
3, // C5 CMP $ab
242+
5, // C6 DEC $ab
243+
0, // C7 DCM* $ab
244+
2, // C8 INY
245+
2, // C9 CMP #$ab
246+
2, // CA DEX
247+
2, // CB SBX* #$ab
248+
4, // CC CPY $abcd
249+
4, // CD CMP $abcd
250+
6, // CE DEC $abcd
251+
0, // CF DCM* $abcd
252+
2, // D0 BNE nearlabel
253+
5, // D1 CMP ($ab),Y
254+
0, // D2 HLT*
255+
0, // D3 DCM* ($ab),Y
256+
0, // D4 SKB* $ab,X
257+
4, // D5 CMP $ab,X
258+
6, // D6 DEC $ab,X
259+
0, // D7 DCM* $ab,X
260+
2, // D8 CLD
261+
4, // D9 CMP $abcd,Y
262+
0, // DA NOP*
263+
0, // DB DCM* $abcd,Y
264+
0, // DC SKW* $abcd,X
265+
4, // DD CMP $abcd,X
266+
7, // DE DEC $abcd,X
267+
0, // DF DCM* $abcd,X
268+
2, // E0 CPX #$ab
269+
6, // E1 SBC ($ab,X)
270+
0, // E2 SKB* #$ab
271+
0, // E3 INS* ($ab,X)
272+
3, // E4 CPX $ab
273+
3, // E5 SBC $ab
274+
5, // E6 INC $ab
275+
0, // E7 INS* $ab
276+
2, // E8 INX
277+
2, // E9 SBC #$ab
278+
2, // EA NOP
279+
0, // EB SBC* #$ab
280+
4, // EC CPX $abcd
281+
4, // ED SBC $abcd
282+
6, // EE INC $abcd
283+
0, // EF INS* $abcd
284+
2, // F0 BEQ nearlabel
285+
5, // F1 SBC ($ab),Y
286+
0, // F2 HLT*
287+
0, // F3 INS* ($ab),Y
288+
0, // F4 SKB* $ab,X
289+
4, // F5 SBC $ab,X
290+
6, // F6 INC $ab,X
291+
0, // F7 INS* $ab,X
292+
2, // F8 SED
293+
4, // F9 SBC $abcd,Y
294+
0, // FA NOP*
295+
0, // FB INS* $abcd,Y
296+
0, // FC SKW* $abcd,X
297+
4, // FD SBC $abcd,X
298+
7, // FE INC $abcd,X
299+
0, // FF INS* $abcd,X
300+
];
301+
302+
#[test]
303+
fn opcode_timing() {
304+
let mut cpu = setup_cpu();
305+
for opcode in 0..256 {
306+
let cycles = OPCODE_TIMING[opcode];
307+
if cycles > 0 {
308+
let clock = Rc::new(Cell::new(0u8));
309+
let clock_clone = clock.clone();
310+
let tick_fn: TickFn = Rc::new(move || {
311+
clock_clone.set(clock_clone.get().wrapping_add(1));
312+
});
313+
cpu.write(0x1000, opcode as u8);
314+
cpu.write(0x1001, 0x00);
315+
cpu.write(0x1002, 0x10);
316+
cpu.set_pc(0x1000);
317+
cpu.step(&tick_fn);
318+
assert_eq!(
319+
cycles,
320+
clock.get(),
321+
"opcode {:02x} timing failed",
322+
opcode as u8
323+
);
324+
}
325+
}
326+
}

‎zinc64-core/tests/cpu_timing.rs

+37-17
Original file line numberDiff line numberDiff line change
@@ -57,7 +57,7 @@ const OPCODE_TIMING: [u8; 256] = [
5757
4, // 0D ORA $abcd
5858
6, // 0E ASL $abcd
5959
0, // 0F ASO* $abcd
60-
2, // 10 BPL nearlabel
60+
3, // 10 BPL nearlabel
6161
5, // 11 ORA ($ab),Y
6262
0, // 12 HLT*
6363
0, // 13 ASO* ($ab),Y
@@ -108,35 +108,35 @@ const OPCODE_TIMING: [u8; 256] = [
108108
6, // 40 RTI
109109
6, // 41 EOR ($ab,X)
110110
0, // 42 HLT*
111-
8, // 43 LSE* ($ab,X)
111+
0, // FIXME 8, // 43 LSE* ($ab,X)
112112
0, // 44 SKB* $ab
113113
3, // 45 EOR $ab
114114
5, // 46 LSR $ab
115-
5, // 47 LSE* $ab
115+
0, // FIXME 5, // 47 LSE* $ab
116116
3, // 48 PHA
117117
2, // 49 EOR #$ab
118118
2, // 4A LSR A
119119
2, // 4B ALR* #$ab
120120
3, // 4C JMP $abcd
121121
4, // 4D EOR $abcd
122122
6, // 4E LSR $abcd
123-
6, // 4F LSE* $abcd
124-
2, // 50 BVC nearlabel
123+
0, // FIXME 6, // 4F LSE* $abcd
124+
3, // 50 BVC nearlabel
125125
5, // 51 EOR ($ab),Y
126126
0, // 52 HLT*
127-
8, // 53 LSE* ($ab),Y
127+
0, // FIXME 8, // 53 LSE* ($ab),Y
128128
0, // 54 SKB* $ab,X
129129
4, // 55 EOR $ab,X
130130
6, // 56 LSR $ab,X
131-
6, // 57 LSE* $ab,X
131+
0, // FIXME 6, // 57 LSE* $ab,X
132132
2, // 58 CLI
133133
4, // 59 EOR $abcd,Y
134134
0, // 5A NOP*
135-
7, // 5B LSE* $abcd,Y
135+
0, // FIXME 7, // 5B LSE* $abcd,Y
136136
0, // 5C SKW* $abcd,X
137137
4, // 5D EOR $abcd,X
138138
7, // 5E LSR $abcd,X
139-
7, // 5F LSE* $abcd,X
139+
0, // FIXME 7, // 5F LSE* $abcd,X
140140
6, // 60 RTS
141141
6, // 61 ADC ($ab,X)
142142
0, // 62 HLT*
@@ -169,7 +169,7 @@ const OPCODE_TIMING: [u8; 256] = [
169169
4, // 7D ADC $abcd,X
170170
7, // 7E ROR $abcd,X
171171
0, // 7F RRA* $abcd,X
172-
3, // FIXME 80 SKB* #$ab
172+
0, // FIXME 3, // FIXME 80 SKB* #$ab
173173
6, // 81 STA ($ab,X)
174174
0, // 82 SKB* #$ab
175175
0, // 83 SAX* ($ab,X)
@@ -185,7 +185,7 @@ const OPCODE_TIMING: [u8; 256] = [
185185
4, // 8D STA $abcd
186186
4, // 8E STX $abcd
187187
0, // 8F SAX* $abcd
188-
2, // 90 BCC nearlabel
188+
3, // 90 BCC nearlabel
189189
6, // 91 STA ($ab),Y
190190
0, // 92 HLT*
191191
0, // 93 SHA* ($ab),Y
@@ -249,7 +249,7 @@ const OPCODE_TIMING: [u8; 256] = [
249249
4, // CD CMP $abcd
250250
6, // CE DEC $abcd
251251
0, // CF DCM* $abcd
252-
2, // D0 BNE nearlabel
252+
3, // D0 BNE nearlabel
253253
5, // D1 CMP ($ab),Y
254254
0, // D2 HLT*
255255
0, // D3 DCM* ($ab),Y
@@ -302,18 +302,38 @@ const OPCODE_TIMING: [u8; 256] = [
302302
#[test]
303303
fn opcode_timing() {
304304
let mut cpu = setup_cpu();
305-
for opcode in 0..256 {
306-
let cycles = OPCODE_TIMING[opcode];
305+
for opcode in 1..=255u8 {
306+
//let opcode = 0x60 as u8;
307+
let cycles = OPCODE_TIMING[opcode as usize];
307308
if cycles > 0 {
308309
let clock = Rc::new(Cell::new(0u8));
309310
let clock_clone = clock.clone();
310311
let tick_fn: TickFn = Rc::new(move || {
311312
clock_clone.set(clock_clone.get().wrapping_add(1));
312313
});
313-
cpu.write(0x1000, opcode as u8);
314-
cpu.write(0x1001, 0x00);
315-
cpu.write(0x1002, 0x10);
314+
cpu.reset();
315+
cpu.write_mem(0x1000, opcode as u8);
316+
cpu.write_mem(0x1001, 0x00);
317+
cpu.write_mem(0x2000, 0x00);
318+
cpu.write_mem(0x2001, 0x10);
319+
cpu.write_mem(0x2100, 0x00);
320+
cpu.write_mem(0x2101, 0x10);
321+
if opcode != 0x20 {
322+
cpu.write_mem(0x1002, 0x20);
323+
} else {
324+
cpu.write_mem(0x1002, 0x21);
325+
}
326+
if opcode == 0x60 {
327+
cpu.write_mem(0x0101, 0x00);
328+
cpu.write_mem(0x0102, 0x10);
329+
}
330+
if opcode == 0x40 {
331+
cpu.write_mem(0x0101, 0x00);
332+
cpu.write_mem(0x0102, 0x01);
333+
cpu.write_mem(0x0103, 0x10);
334+
}
316335
cpu.set_pc(0x1000);
336+
cpu.clock();
317337
cpu.step(&tick_fn);
318338
assert_eq!(
319339
cycles,

‎zinc64-debug/src/debugger.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,8 @@ use std::u8;
1313

1414
use bit_field::BitField;
1515
use byteorder::{BigEndian, ReadBytesExt};
16-
use zinc64_core::cpu::Instruction;
16+
17+
use crate::instruction::Instruction;
1718

1819
use super::charset;
1920
use super::disassembler::Disassembler;

‎zinc64-debug/src/disassembler.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44

55
#![cfg_attr(feature = "cargo-clippy", allow(clippy::cast_lossless))]
66

7-
use zinc64_core::cpu::{Instruction, Operand};
7+
use crate::instruction::{Instruction, Operand};
88

99
pub struct Disassembler {
1010
data: Vec<u8>,

‎zinc64-debug/src/instruction.rs

+188
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,188 @@
1+
// This file is part of zinc64.
2+
// Copyright (c) 2016-2019 Sebastian Jastrzebski. All rights reserved.
3+
// Licensed under the GPLv3. See LICENSE file in the project root for full license text.
4+
#![allow(dead_code)]
5+
6+
use core::fmt;
7+
8+
pub enum Operand {
9+
Accumulator,
10+
Immediate(u8),
11+
ZeroPage(u8),
12+
ZeroPageX(u8),
13+
ZeroPageY(u8),
14+
Absolute(u16),
15+
AbsoluteX(u16),
16+
AbsoluteY(u16),
17+
IndirectX(u8),
18+
IndirectY(u8),
19+
Indirect(u16),
20+
Relative(i8),
21+
}
22+
23+
pub enum Instruction {
24+
// Data Movement (16)
25+
LDA(Operand),
26+
LDX(Operand),
27+
LDY(Operand),
28+
PHA,
29+
PHP,
30+
PLA,
31+
PLP,
32+
STA(Operand),
33+
STX(Operand),
34+
STY(Operand),
35+
TAX,
36+
TAY,
37+
TSX,
38+
TXA,
39+
TXS,
40+
TYA,
41+
// Arithmetic (11)
42+
ADC(Operand),
43+
SBC(Operand),
44+
CMP(Operand),
45+
CPX(Operand),
46+
CPY(Operand),
47+
DEC(Operand),
48+
DEX,
49+
DEY,
50+
INC(Operand),
51+
INX,
52+
INY,
53+
// Logical (3)
54+
AND(Operand),
55+
EOR(Operand),
56+
ORA(Operand),
57+
// Shift and Rotate (4)
58+
ASL(Operand),
59+
LSR(Operand),
60+
ROL(Operand),
61+
ROR(Operand),
62+
// Control Flow (11)
63+
BCC(Operand),
64+
BCS(Operand),
65+
BEQ(Operand),
66+
BMI(Operand),
67+
BNE(Operand),
68+
BPL(Operand),
69+
BVC(Operand),
70+
BVS(Operand),
71+
JMP(Operand),
72+
JSR(Operand),
73+
RTS,
74+
// Misc (11)
75+
BIT(Operand),
76+
BRK,
77+
CLC,
78+
CLD,
79+
CLI,
80+
CLV,
81+
NOP,
82+
SEC,
83+
SED,
84+
SEI,
85+
RTI,
86+
// Undocumented
87+
ANE(Operand),
88+
ANX(Operand),
89+
ALR(Operand),
90+
AXS(Operand),
91+
LAX(Operand),
92+
LSE(Operand),
93+
}
94+
95+
impl fmt::Display for Operand {
96+
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
97+
match *self {
98+
Operand::Accumulator => write!(f, "acc"),
99+
Operand::Immediate(value) => write!(f, "#{:02x}", value),
100+
Operand::ZeroPage(address) => write!(f, "${:02x}", address),
101+
Operand::ZeroPageX(address) => write!(f, "${:02x},x", address),
102+
Operand::ZeroPageY(address) => write!(f, "${:02x},y", address),
103+
Operand::Absolute(address) => write!(f, "${:04x}", address),
104+
Operand::AbsoluteX(address) => write!(f, "${:04x},x", address),
105+
Operand::AbsoluteY(address) => write!(f, "${:04x},y", address),
106+
Operand::IndirectX(address) => write!(f, "$({:02x},x)", address),
107+
Operand::IndirectY(address) => write!(f, "$({:02x},y)", address),
108+
Operand::Indirect(address) => write!(f, "$({:04x})", address),
109+
Operand::Relative(offset) => write!(f, "${:02x}", offset),
110+
}
111+
}
112+
}
113+
114+
impl fmt::Display for Instruction {
115+
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
116+
match *self {
117+
// Data Movement
118+
Instruction::LDA(ref operand) => write!(f, "lda {}", operand),
119+
Instruction::LDX(ref operand) => write!(f, "ldx {}", operand),
120+
Instruction::LDY(ref operand) => write!(f, "ldy {}", operand),
121+
Instruction::PHA => write!(f, "pha"),
122+
Instruction::PHP => write!(f, "php"),
123+
Instruction::PLA => write!(f, "pla"),
124+
Instruction::PLP => write!(f, "plp"),
125+
Instruction::STA(ref operand) => write!(f, "sta {}", operand),
126+
Instruction::STX(ref operand) => write!(f, "stx {}", operand),
127+
Instruction::STY(ref operand) => write!(f, "sty {}", operand),
128+
Instruction::TAX => write!(f, "tax"),
129+
Instruction::TAY => write!(f, "tay"),
130+
Instruction::TSX => write!(f, "tsx"),
131+
Instruction::TXA => write!(f, "txa"),
132+
Instruction::TXS => write!(f, "txs"),
133+
Instruction::TYA => write!(f, "tya"),
134+
// Arithmetic
135+
Instruction::ADC(ref operand) => write!(f, "adc {}", operand),
136+
Instruction::SBC(ref operand) => write!(f, "sbc {}", operand),
137+
Instruction::CMP(ref operand) => write!(f, "cmp {}", operand),
138+
Instruction::CPX(ref operand) => write!(f, "cpx {}", operand),
139+
Instruction::CPY(ref operand) => write!(f, "cpy {}", operand),
140+
Instruction::DEC(ref operand) => write!(f, "dec {}", operand),
141+
Instruction::DEX => write!(f, "dex"),
142+
Instruction::DEY => write!(f, "dey"),
143+
Instruction::INC(ref operand) => write!(f, "inc {}", operand),
144+
Instruction::INX => write!(f, "inx"),
145+
Instruction::INY => write!(f, "iny"),
146+
// Logical
147+
Instruction::AND(ref operand) => write!(f, "and {}", operand),
148+
Instruction::EOR(ref operand) => write!(f, "eor {}", operand),
149+
Instruction::ORA(ref operand) => write!(f, "ora {}", operand),
150+
// Shift and Rotate
151+
Instruction::ASL(ref operand) => write!(f, "asl {}", operand),
152+
Instruction::LSR(ref operand) => write!(f, "lsr {}", operand),
153+
Instruction::ROL(ref operand) => write!(f, "rol {}", operand),
154+
Instruction::ROR(ref operand) => write!(f, "ror {}", operand),
155+
// Control Flow
156+
Instruction::BCC(ref operand) => write!(f, "bcc {}", operand),
157+
Instruction::BCS(ref operand) => write!(f, "bcs {}", operand),
158+
Instruction::BEQ(ref operand) => write!(f, "beq {}", operand),
159+
Instruction::BMI(ref operand) => write!(f, "bmi {}", operand),
160+
Instruction::BNE(ref operand) => write!(f, "bne {}", operand),
161+
Instruction::BPL(ref operand) => write!(f, "bpl {}", operand),
162+
Instruction::BVC(ref operand) => write!(f, "bvc {}", operand),
163+
Instruction::BVS(ref operand) => write!(f, "bvs {}", operand),
164+
Instruction::JMP(ref operand) => write!(f, "jmp {}", operand),
165+
Instruction::JSR(ref operand) => write!(f, "jsr {}", operand),
166+
Instruction::RTS => write!(f, "rts"),
167+
// Misc
168+
Instruction::BIT(ref operand) => write!(f, "bit {}", operand),
169+
Instruction::BRK => write!(f, "brk"),
170+
Instruction::CLC => write!(f, "clc"),
171+
Instruction::CLD => write!(f, "cld"),
172+
Instruction::CLI => write!(f, "cli"),
173+
Instruction::CLV => write!(f, "clv"),
174+
Instruction::NOP => write!(f, "nop"),
175+
Instruction::SEC => write!(f, "sec"),
176+
Instruction::SED => write!(f, "sed"),
177+
Instruction::SEI => write!(f, "sei"),
178+
Instruction::RTI => write!(f, "rti"),
179+
// Undocumented
180+
Instruction::ANE(ref operand) => write!(f, "ane {}", operand),
181+
Instruction::ANX(ref operand) => write!(f, "anx {}", operand),
182+
Instruction::ALR(ref operand) => write!(f, "alr {}", operand),
183+
Instruction::AXS(ref operand) => write!(f, "axs {}", operand),
184+
Instruction::LAX(ref operand) => write!(f, "lax {}", operand),
185+
Instruction::LSE(ref operand) => write!(f, "lse {}", operand),
186+
}
187+
}
188+
}

‎zinc64-debug/src/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ extern crate log;
88
mod charset;
99
mod debugger;
1010
mod disassembler;
11+
mod instruction;
1112
mod rap_server;
1213

1314
use std::sync::mpsc::Sender;

‎zinc64-system/src/c64.rs

+2-6
Original file line numberDiff line numberDiff line change
@@ -69,7 +69,6 @@ pub struct C64 {
6969
breakpoints: BreakpointManager,
7070
clock: Rc<Clock>,
7171
frame_count: u32,
72-
last_pc: u16,
7372
tick_fn: TickFn,
7473
vsync_flag: SharedCell<bool>,
7574
}
@@ -246,7 +245,6 @@ impl C64 {
246245
breakpoints: BreakpointManager::default(),
247246
clock,
248247
frame_count: 0,
249-
last_pc: 0,
250248
tick_fn,
251249
vsync_flag,
252250
}
@@ -329,7 +327,7 @@ impl C64 {
329327
}
330328

331329
pub fn is_cpu_jam(&self) -> bool {
332-
self.last_pc == self.cpu.get_pc()
330+
self.cpu.is_cpu_jam()
333331
}
334332

335333
pub fn set_autostart(&mut self, autostart: Option<Autostart>) {
@@ -386,7 +384,6 @@ impl C64 {
386384
self.sound_buffer.reset();
387385
// Runtime State
388386
self.frame_count = 0;
389-
self.last_pc = 0;
390387
self.vsync_flag.set(false);
391388
}
392389

@@ -421,9 +418,8 @@ impl C64 {
421418

422419
#[inline]
423420
pub fn step_internal(&mut self, tick_fn: &TickFn) {
424-
self.last_pc = self.cpu.get_pc();
425421
self.cpu.step(&tick_fn);
426-
if self.autostart.is_some() && self.cpu.get_pc() == BaseAddr::BootComplete.addr() {
422+
if self.autostart.is_some() && self.cpu.get_pc() == (BaseAddr::BootComplete.addr()) {
427423
if let Some(mut autostart) = self.autostart.take() {
428424
autostart.execute(self);
429425
}

‎zinc64-system/src/condition.rs

+6-6
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ use alloc::prelude::*;
99
use core::fmt;
1010
use core::iter::Peekable;
1111
use core::str::Chars;
12-
use zinc64_core::factory::Cpu;
12+
use zinc64_core::factory::{Cpu, Register};
1313

1414
enum Operator {
1515
Equal,
@@ -103,11 +103,11 @@ impl Condition {
103103

104104
fn eval_reg(&self, reg: &Reg, cpu: &dyn Cpu) -> u16 {
105105
match *reg {
106-
Reg::A => cpu.get_a() as u16,
107-
Reg::X => cpu.get_x() as u16,
108-
Reg::Y => cpu.get_y() as u16,
109-
Reg::P => cpu.get_p() as u16,
110-
Reg::SP => cpu.get_sp() as u16,
106+
Reg::A => cpu.get_register(Register::A) as u16,
107+
Reg::X => cpu.get_register(Register::X) as u16,
108+
Reg::Y => cpu.get_register(Register::Y) as u16,
109+
Reg::P => cpu.get_register(Register::P) as u16,
110+
Reg::SP => cpu.get_register(Register::SP) as u16,
111111
Reg::PC => cpu.get_pc(),
112112
}
113113
}

‎zinc64/src/debug.rs

+11-10
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ use std::sync::mpsc::Sender;
1010
use std::time::Duration;
1111

1212
use byteorder::{BigEndian, WriteBytesExt};
13+
use zinc64_core::factory::Register;
1314
use zinc64_debug::{Command, Output, RegData, RegOp};
1415
use zinc64_system::C64;
1516

@@ -242,11 +243,11 @@ impl Debug {
242243
let clock = c64.get_clock().get();
243244
let cpu = c64.get_cpu();
244245
let regs = RegData {
245-
a: cpu.get_a(),
246-
x: cpu.get_x(),
247-
y: cpu.get_y(),
248-
p: cpu.get_p(),
249-
sp: cpu.get_sp(),
246+
a: cpu.get_register(Register::A),
247+
x: cpu.get_register(Register::X),
248+
y: cpu.get_register(Register::Y),
249+
p: cpu.get_register(Register::P),
250+
sp: cpu.get_register(Register::SP),
250251
pc: cpu.get_pc(),
251252
port_00: cpu.read(0x00),
252253
port_01: cpu.read(0x01),
@@ -259,11 +260,11 @@ impl Debug {
259260
let cpu = c64.get_cpu_mut();
260261
for op in ops {
261262
match *op {
262-
RegOp::SetA(value) => cpu.set_a(value),
263-
RegOp::SetX(value) => cpu.set_x(value),
264-
RegOp::SetY(value) => cpu.set_y(value),
265-
RegOp::SetP(value) => cpu.set_p(value),
266-
RegOp::SetSP(value) => cpu.set_sp(value),
263+
RegOp::SetA(value) => cpu.set_register(Register::A, value),
264+
RegOp::SetX(value) => cpu.set_register(Register::X, value),
265+
RegOp::SetY(value) => cpu.set_register(Register::Y, value),
266+
RegOp::SetP(value) => cpu.set_register(Register::P, value),
267+
RegOp::SetSP(value) => cpu.set_register(Register::SP, value),
267268
RegOp::SetPC(value) => cpu.set_pc(value),
268269
}
269270
}

0 commit comments

Comments
 (0)
Please sign in to comment.