Calculate and verify ROM checksum

This commit is contained in:
Michael Smith
2025-08-13 13:48:38 +02:00
parent ba4e098ba5
commit c3d17459c6
4 changed files with 32 additions and 326 deletions

View File

@@ -3,8 +3,7 @@ package gb
import (
"bytes"
"encoding/binary"
"io"
"log"
"fmt"
"os"
)
@@ -215,6 +214,7 @@ var oldLicensees = map[byte]string{
}
type ROMHeader struct {
_ [256]byte
EntryPoint [4]byte
Logo [48]byte
// NOTE(m): Assuming "old" cartridges here. This may cause problems with newer cartridges.
@@ -228,7 +228,7 @@ type ROMHeader struct {
DestinationCode byte
OldLicenseeCode byte
MaskROMVersionNumber byte
HeaderChecksum byte
Checksum byte
GlobalChecksum [2]byte
}
@@ -242,32 +242,23 @@ type Cartridge struct {
RAMSize string
Destination string
Version int
Checksum byte
}
func InsertCartridge(path string) *Cartridge {
file, err := os.Open(path)
if err != nil {
log.Fatal(err)
}
defer file.Close()
cartridge := Cartridge{Filename: file.Name()}
func InsertCartridge(filename string) (*Cartridge, error) {
cartridge := Cartridge{Filename: filename}
// Jump to start of header
_, err = file.Seek(0x0100, io.SeekStart)
rom, err := os.ReadFile(filename)
if err != nil {
log.Fatal(err)
return &cartridge, err
}
// Read header
var header ROMHeader
err = binary.Read(file, binary.LittleEndian, &header)
buffer := bytes.NewReader(rom)
err = binary.Read(buffer, binary.LittleEndian, &header)
if err != nil {
log.Fatal(err)
}
// Validate the ROM by checking presence of the Nintendo logo
if header.Logo != expectedLogo {
log.Fatal("Invalid ROM file: No valid logo found!")
return &cartridge, nil
}
// Convert some header values
@@ -292,10 +283,16 @@ func InsertCartridge(path string) *Cartridge {
}
cartridge.Version = int(header.MaskROMVersionNumber)
// TODO(m): Verify header checksum
// Calculate and verify checksum
for address := uint16(0x0134); address <= uint16(0x014C); address++ {
cartridge.Checksum = cartridge.Checksum - rom[address] - 1
}
if cartridge.Checksum != header.Checksum {
return &cartridge, fmt.Errorf("ROM checksum failed: %X does not equal %X", cartridge.Checksum, header.Checksum)
}
// NOTE(m): Ignoring global checksum which is not used, except by one emulator.
// See https://gbdev.io/pandocs/The_Cartridge_Header.html#014e-014f--global-checksum
return &cartridge
return &cartridge, nil
}

31
gb/cartridge_test.go Normal file
View File

@@ -0,0 +1,31 @@
package gb
import (
"testing"
"github.com/stretchr/testify/assert"
)
func TestInsert(t *testing.T) {
cartridge, err := InsertCartridge("../roms/dmg-acid2.gb")
assert := assert.New(t)
assert.Nil(err)
assert.Equal(cartridge.Filename, "../roms/dmg-acid2.gb")
assert.Equal(cartridge.Title, "DMG-ACID2")
assert.Equal(cartridge.Mapper, "ROM ONLY")
assert.Equal(cartridge.Licensee, "None")
assert.False(cartridge.SGBSupport, "SGB support should be false")
assert.Equal(cartridge.ROMSize, 32)
assert.Equal(cartridge.RAMSize, "0 - No RAM")
assert.Equal(cartridge.Destination, "Japan (and possibly overseas)")
assert.Equal(cartridge.Version, 0)
assert.Equal(cartridge.Checksum, byte(0x9F))
}
func TestFailedChecksum(t *testing.T) {
_, err := InsertCartridge("../roms/failed-checksum.gb")
assert.EqualError(t, err, "ROM checksum failed: 9F does not equal 41")
}