1
0
mirror of https://github.com/zellyn/go6502.git synced 2025-01-03 19:30:50 +00:00
go6502/tests/functional_test.go

160 lines
3.3 KiB
Go

/*
Tests for the 6502 CPU emulator.
*/
package tests
import (
"fmt"
"io/ioutil"
"math/rand"
"testing"
"github.com/zellyn/go6502/asm"
"github.com/zellyn/go6502/cpu"
)
// Memory for the tests. Satisfies the cpu.Memory interface.
type K64 [65536]byte
func (m *K64) Read(address uint16) byte {
return m[address]
}
func (m *K64) Write(address uint16, value byte) {
m[address] = value
}
// Cycle counter for the tests. Satisfies the cpu.Ticker interface.
type CycleCount uint64
func (c *CycleCount) Tick() {
*c += 1
}
func randomize(k *K64) {
for i := 0; i < 65536; i++ {
k[i] = byte(rand.Int())
}
}
// printStatus prints out the current CPU instruction and register status.
func printStatus(c cpu.Cpu, m K64, cc CycleCount) {
bytes, text, _ := asm.Disasm(c.PC(), m[c.PC()], m[c.PC()+1], m[c.PC()+2])
fmt.Printf("$%04X: %s %s A=$%02X X=$%02X Y=$%02X SP=$%02X P=$%08b - %d\n",
c.PC(), bytes, text, c.A(), c.X(), c.Y(), c.SP(), c.P(), cc)
}
// Run Klaus Dormann's amazing comprehensive test.
func TestFunctionalTest(t *testing.T) {
unused := map[byte]bool{}
for k, _ := range cpu.Opcodes {
unused[k] = true
}
bytes, err := ioutil.ReadFile("6502_functional_test.bin")
if err != nil {
panic("Cannot read file")
}
var m K64
randomize(&m)
var cc CycleCount
OFFSET := 0xa
copy(m[OFFSET:len(bytes)+OFFSET], bytes)
c := cpu.NewCPU(&m, &cc, cpu.VERSION_6502)
c.Reset()
c.SetPC(0x1000)
for {
unused[m[c.PC()]] = false
oldPC := c.PC()
// printStatus(c, m, cc)
err := c.Step()
if err != nil {
t.Error(err)
break
}
if c.PC() == oldPC {
if c.PC() != 0x3BB5 {
t.Errorf("Stuck at $%04X: 0x%02X", oldPC, m[oldPC])
}
break
}
}
for k, v := range unused {
if v {
t.Errorf("Unused instruction: 0x%2X", k)
}
}
}
// Run Bruce Clark's decimal test in 6502 mode.
func TestDecimalMode6502(t *testing.T) {
bytes, err := ioutil.ReadFile("decimal_mode.bin")
if err != nil {
panic("Cannot read file")
}
var m K64
randomize(&m)
var cc CycleCount
OFFSET := 0x1000
copy(m[OFFSET:len(bytes)+OFFSET], bytes)
m[1] = 0 // 6502
c := cpu.NewCPU(&m, &cc, cpu.VERSION_6502)
c.Reset()
c.SetPC(0x1000)
for {
oldPC := c.PC()
// printStatus(c, m, cc)
err := c.Step()
if err != nil {
t.Error(err)
break
}
if c.PC() == oldPC {
if c.PC() != 0x1037 {
t.Errorf("Stuck at 0x%X: 0x%X\n", oldPC, m[oldPC])
}
break
}
}
error := m[0]
if error > 0 {
t.Errorf("Decimal mode test failed: error=%d", error)
}
}
// Run Bruce Clark's decimal test in 65C02 mode.
func TestDecimalMode65C02(t *testing.T) {
bytes, err := ioutil.ReadFile("decimal_mode.bin")
if err != nil {
panic("Cannot read file")
}
var m K64
randomize(&m)
var cc CycleCount
OFFSET := 0x1000
copy(m[OFFSET:len(bytes)+OFFSET], bytes)
m[1] = 1 // 65C02
c := cpu.NewCPU(&m, &cc, cpu.VERSION_65C02)
c.Reset()
c.SetPC(0x1000)
for {
oldPC := c.PC()
// printStatus(c, m, cc)
err := c.Step()
if err != nil {
t.Error(err)
break
}
if c.PC() == oldPC {
if c.PC() != 0x1037 {
t.Errorf("Stuck at 0x%X: 0x%X\n", oldPC, m[oldPC])
}
break
}
}
error := m[0]
if error > 0 {
fmt.Printf("N1=$%02X N2=$%02X DA=$%02X DNVZC=$%02X - AR=$%02X NF=$%02X VF=$%02X ZF=$%02X CF=$%02X\n",
m[0x08], m[0x0B], m[0x04], m[0x05], m[0x02], m[0x0D], m[0x0E], m[0x0F], m[0x03])
t.Errorf("Decimal mode test failed: error=%d", error)
}
}