Gopher2600/hardware/memory/cartridge/fingerprint.go
JetSetIlly 1f23e7217f simplified cartridge loader package
loader is opened at creation time which allows us to remove the
needlessly complicated pointer-to-pointer mechanism
2024-04-16 10:18:13 +01:00

520 lines
14 KiB
Go

// This file is part of Gopher2600.
//
// Gopher2600 is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// Gopher2600 is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with Gopher2600. If not, see <https://www.gnu.org/licenses/>.
package cartridge
import (
"bytes"
"fmt"
"github.com/jetsetilly/gopher2600/cartridgeloader"
"github.com/jetsetilly/gopher2600/environment"
"github.com/jetsetilly/gopher2600/hardware/memory/cartridge/ace"
"github.com/jetsetilly/gopher2600/hardware/memory/cartridge/cdf"
"github.com/jetsetilly/gopher2600/hardware/memory/cartridge/dpcplus"
"github.com/jetsetilly/gopher2600/hardware/memory/cartridge/elf"
"github.com/jetsetilly/gopher2600/hardware/memory/cartridge/mapper"
"github.com/jetsetilly/gopher2600/hardware/memory/cartridge/supercharger"
)
// if anwhere parameter is true then the ELF magic number can appear anywhere
// in the data (the b parameter). otherwise it must appear at the beginning of
// the data
func fingerprintElf(b []byte, anywhere bool) bool {
if anywhere {
if bytes.Contains(b, []byte{0x7f, 'E', 'L', 'F'}) {
return true
}
} else if bytes.HasPrefix(b, []byte{0x7f, 'E', 'L', 'F'}) {
return true
}
return false
}
func fingerprintAce(b []byte) (bool, bool) {
if len(b) < 144 {
return false, false
}
// some ACE files embed an ELF file inside the ACE data. these files are
// identified by the presence of "elf-relocatable" in the data premable
wrappedELF := bytes.Contains(b[:144], []byte("elf-relocatable"))
// make double sure this is actually an elf file. otherwise it's just an
// ACE file with elf-relocatable in the data preamble
wrappedELF = wrappedELF && fingerprintElf(b, true)
if bytes.Contains(b[:144], []byte("ACE-2600")) {
return true, wrappedELF
}
if bytes.Contains(b[:144], []byte("ACE-PC00")) {
return true, wrappedELF
}
if bytes.Contains(b[:144], []byte("ACE-UF00")) {
return true, wrappedELF
}
return false, false
}
func fingerprint3e(b []byte) bool {
// 3E cart bankswitching is triggered by storing the bank number in address
// 3E using 'STA $3E', commonly followed by an immediate mode LDA
//
// fingerprint method taken from:
//
// https://gitlab.com/firmaplus/atari-2600-pluscart/-/blob/master/source/STM32firmware/PlusCart/Src/cartridge_detection.c#L140
for i := 0; i < len(b)-3; i++ {
if b[i] == 0x85 && b[i+1] == 0x3e && b[i+2] == 0xa9 && b[i+3] == 0x00 {
return true
}
}
return false
}
func fingerprint3ePlus(b []byte) bool {
// previous versions of this function worked similarly to the tigervision
// method but this is more accurate
//
// fingerprint method taken from:
//
// https://gitlab.com/firmaplus/atari-2600-pluscart/-/blob/master/source/STM32firmware/PlusCart/Src/cartridge_detection.c#L148
for i := 0; i < len(b)-3; i++ {
if b[i] == 'T' && b[i+1] == 'J' && b[i+2] == '3' && b[i+3] == 'E' {
return true
}
}
return false
}
func fingerprintMnetwork(b []byte) bool {
// Bump 'n' Jump is the fussiest mnetwork cartridge I've found. Matching
// hotspots:
//
// $fdd5 LDA BANK5
// $fde3 LDA BANK6
// $fe0e LDA BANK5
// $fe16 LDA BANK4
//
// This also catches modern games not created by mnetwork but which use the
// format, eg Pitkat
//
// (24/01/21)
//
// Splendidnut's Congobongo demo is even fussier than Bump 'n' Jump. While
// I expect the ROM to get more complex we should support the demo ROM if
// only because it exists.
//
// notably, it uses a non-primary-mirror cartridge address and there are
// only two bankswitch instructions in the data (although I didn't search
// it exhaustively - just LDA instructions).
//
// threshold has been reduced to two.
//
// (19/06/21)
//
// Thomas Jentzch's Elite demo requires a threshold of one
//
// https://atariage.com/forums/topic/155657-elite-3d-graphics/?do=findComment&comment=2444328
//
// with such a low threshold, mnetwork should probably be the very last
// type to check for
threshold := 1
for i := 0; i < len(b)-3; i++ {
if b[i] == 0xad && (b[i+1] >= 0xe0 && b[i+1] <= 0xe7) {
// bank switching can address any cartidge mirror so mask off
// insignificant bytes
//
// (09/03/21) mask wasn't correct (0x0f selects non-cartridge
// mirrors too) correct mask is 0x1f.
//
// the incorrect mask caused a false positive for Solaris when the
// threshold is 2.
//
// (20/06/21) this caused a falso positive for "Hack Em Hangly Pacman"
// when the threshold is 1
//
// change to only look for mirrors 0x1f and 0xff
if b[i+2] == 0x1f || b[i+2] == 0xff {
threshold--
if threshold == 0 {
return true
}
}
}
}
return false
}
func fingerprintParkerBros(b []byte) bool {
// parker bros fingerprint taken from Stella
fingerprint := [][]byte{
{0x8d, 0xe0, 0x1f}, // STA $1FE0
{0x8d, 0xe0, 0x5f}, // STA $5FE0
{0x8d, 0xe9, 0xff}, // STA $FFE9
{0x0c, 0xe0, 0x1f}, // NOP $1FE0
{0xad, 0xe0, 0x1f}, // LDA $1FE0
{0xad, 0xe9, 0xff}, // LDA $FFE9
{0xad, 0xed, 0xff}, // LDA $FFED
{0xad, 0xf3, 0xbf}, // LDA $BFF3
}
for _, f := range fingerprint {
if bytes.Contains(b, f) {
return true
}
}
return false
}
func fingerprintDF(b []byte) bool {
if len(b) < 0xffb {
return false
}
return b[0xff8] == 'D' && b[0xff9] == 'F' && b[0xffa] == 'S' && b[0xffb] == 'C'
}
func fingerprintWickstead(b []byte) bool {
// wickstead design fingerprint taken from Stella
return bytes.Contains(b, []byte{0xa5, 0x39, 0x4c})
}
func fingerprintSCABS(b []byte) bool {
// SCABS fingerprint taken from Stella
fingerprint := [][]byte{
{0x20, 0x00, 0xd0, 0xc6, 0xc5}, // JSR $D000; DEC $C5
{0x20, 0xc3, 0xf8, 0xa5, 0x82}, // JSR $F8C3; LDA $82
{0xd0, 0xfB, 0x20, 0x73, 0xfe}, // BNE $FB; JSR $FE73
{0x20, 0x00, 0xf0, 0x84, 0xd6}, // JSR $F000; $84, $D6
}
for _, f := range fingerprint {
if bytes.Contains(b, f) {
return true
}
}
return false
}
func fingerprintUA(b []byte) bool {
// ua fingerprint taken from Stella
fingerprint := [][]byte{
{0x8D, 0x40, 0x02}, // STA $240 (Funky Fish, Pleiades)
{0xAD, 0x40, 0x02}, // LDA $240 (???)
{0xBD, 0x1F, 0x02}, // LDA $21F,X (Gingerbread Man)
{0x2C, 0xC0, 0x02}, // BIT $2C0 (Time Pilot)
{0x8D, 0xC0, 0x02}, // STA $2C0 (Fathom, Vanguard)
{0xAD, 0xC0, 0x02}, // LDA $2C0 (Mickey)
}
for _, f := range fingerprint {
if bytes.Contains(b, f) {
return true
}
}
return false
}
func fingerprintDPCplus(b []byte) bool {
if len(b) < 0x23 {
return false
}
return b[0x20] == 0x1e && b[0x21] == 0xab && b[0x22] == 0xad && b[0x23] == 0x10
}
func fingerprintCDFJplus(b []byte) (bool, string) {
if len(b) < 2048 {
return false, ""
}
if bytes.Contains(b[:2048], []byte("PLUSCDFJ")) {
return true, "CDFJ+"
}
return false, ""
}
func fingerprintCDF(b []byte) (bool, string) {
count := 0
version := ""
for i := 0; i < len(b)-3; i++ {
if b[i] == 'C' && b[i+1] == 'D' && b[i+2] == 'F' {
var newVersion string
count++
// create version string. slightly different for CDFJ
if b[i+3] == 'J' {
newVersion = "CDFJ"
} else {
newVersion = fmt.Sprintf("CDF%1d", b[i+3])
}
// make sure the version number hasn't changed
if version != "" && version != newVersion {
return false, ""
}
version = newVersion
}
}
return count >= 3, version
}
func fingerprintSuperchargerFastLoad(cartload cartridgeloader.Loader) bool {
return len(cartload.Data) > 0 && len(cartload.Data)%8448 == 0
}
func fingerprintTigervision(b []byte) bool {
// tigervision cartridges change banks by writing to memory address 0x3f. we
// can hypothesise that these types of cartridges will have that instruction
// sequence "85 3f" many times in a ROM whereas other cartridge types will not
threshold := 5
for i := 0; i < len(b)-1; i++ {
if b[i] == 0x85 && b[i+1] == 0x3f {
threshold--
}
if threshold == 0 {
return true
}
}
return false
}
func fingerprint8k(data []byte) func(*environment.Environment, []byte) (mapper.CartMapper, error) {
if fingerprintTigervision(data) {
return newTigervision
}
if fingerprintParkerBros(data) {
return newParkerBros
}
// mnetwork has the lowest threshold so place it at the end
if fingerprintMnetwork(data) {
return newMnetwork
}
if fingerprintWickstead(data) {
return newWicksteadDesign
}
if fingerprintSCABS(data) {
return newSCABS
}
if fingerprintUA(data) {
return newUA
}
return newAtari8k
}
func fingerprint16k(data []byte) func(*environment.Environment, []byte) (mapper.CartMapper, error) {
if fingerprintTigervision(data) {
return newTigervision
}
if fingerprintMnetwork(data) {
return newMnetwork
}
return newAtari16k
}
func fingerprint32k(data []byte) func(*environment.Environment, []byte) (mapper.CartMapper, error) {
if fingerprintTigervision(data) {
return newTigervision
}
return newAtari32k
}
func fingerprint64k(data []byte) func(*environment.Environment, []byte) (mapper.CartMapper, error) {
return newEF
}
func fingerprint128k(data []byte) func(*environment.Environment, []byte) (mapper.CartMapper, error) {
if fingerprintDF(data) {
return newDF
}
return newSuperbank
}
func fingerprint256k(data []byte) func(*environment.Environment, []byte) (mapper.CartMapper, error) {
return newSuperbank
}
func (cart *Cartridge) fingerprint(cartload cartridgeloader.Loader) error {
var err error
// moviecart fingerprinting is done in cartridge loader. this is to avoid
// loading the entire file into memory, which we definitely don't want to do
// with moviecart files due to the large size
if ok := fingerprintElf(cartload.Data, false); ok {
cart.mapper, err = elf.NewElf(cart.env, cart.Filename, false)
return err
}
if ok, wrappedElf := fingerprintAce(cartload.Data); ok {
if wrappedElf {
cart.mapper, err = elf.NewElf(cart.env, cart.Filename, true)
return err
}
cart.mapper, err = ace.NewAce(cart.env, cartload.Data)
return err
}
if ok, version := fingerprintCDFJplus(cartload.Data); ok {
cart.mapper, err = cdf.NewCDF(cart.env, version, cartload.Data)
return err
}
if ok, version := fingerprintCDF(cartload.Data); ok {
cart.mapper, err = cdf.NewCDF(cart.env, version, cartload.Data)
return err
}
if fingerprintDPCplus(cartload.Data) {
cart.mapper, err = dpcplus.NewDPCplus(cart.env, cartload.Data)
return err
}
if fingerprintSuperchargerFastLoad(cartload) {
cart.mapper, err = supercharger.NewSupercharger(cart.env, cartload)
return err
}
if fingerprint3ePlus(cartload.Data) {
cart.mapper, err = new3ePlus(cart.env, cartload.Data)
return err
}
if fingerprint3e(cartload.Data) {
cart.mapper, err = new3e(cart.env, cartload.Data)
return err
}
sz := len(cartload.Data)
switch sz {
case 4096:
cart.mapper, err = newAtari4k(cart.env, cartload.Data)
if err != nil {
return err
}
case 8195:
// a widely distributed bad ROM dump of the Pink Panther prototype is
// 8195 bytes long. we'll treat it like an 8k ROM and see if it's
// recognised as a Wickstead Design ROM. if it's not then it's just a
// file that's 8195 bytes long and will be rejected
fallthrough
case 8192:
cart.mapper, err = fingerprint8k(cartload.Data)(cart.env, cartload.Data)
if err != nil {
return err
}
case 10240:
fallthrough
case 10495:
cart.mapper, err = newDPC(cart.env, cartload.Data)
if err != nil {
return err
}
case 12288:
cart.mapper, err = newCBS(cart.env, cartload.Data)
if err != nil {
return err
}
case 16384:
cart.mapper, err = fingerprint16k(cartload.Data)(cart.env, cartload.Data)
if err != nil {
return err
}
case 32768:
cart.mapper, err = fingerprint32k(cartload.Data)(cart.env, cartload.Data)
if err != nil {
return err
}
case 65536:
cart.mapper, err = fingerprint64k(cartload.Data)(cart.env, cartload.Data)
if err != nil {
return err
}
case 131072:
cart.mapper, err = fingerprint128k(cartload.Data)(cart.env, cartload.Data)
if err != nil {
return err
}
case 262144:
cart.mapper, err = fingerprint256k(cartload.Data)(cart.env, cartload.Data)
if err != nil {
return err
}
default:
if sz >= 4096 {
return fmt.Errorf("unrecognised size (%d bytes)", len(cartload.Data))
}
cart.mapper, err = newAtari2k(cart.env, cartload.Data)
if err != nil {
return err
}
}
// if cartridge mapper implements the optionalSuperChip interface then try
// to add the additional RAM
if superchip, ok := cart.mapper.(mapper.OptionalSuperchip); ok {
superchip.AddSuperchip(false)
}
return nil
}
// fingerprinting a PlusROM cartridge is slightly different to the main
// fingerprint() function above. the fingerprintPlusROM() function below is the
// first step. it checks for the byte sequence 8d f1 x1, which is the
// equivalent to STA $xff1, a necessary instruction in a PlusROM cartridge
//
// if this sequence is found then the function returns true, whereupon
// plusrom.NewPlusROM() can be called. the seoncd part of the fingerprinting
// process occurs in that function. if that fails then we can say that the true
// result from this function was a false positive.
func (cart *Cartridge) fingerprintPlusROM(cartload cartridgeloader.Loader) bool {
for i := 0; i < len(cartload.Data)-2; i++ {
if (cartload.Data)[i] == 0x8d && (cartload.Data)[i+1] == 0xf1 && ((cartload.Data)[i+2]&0x10) == 0x10 {
return true
}
}
return false
}