Gopher2600/hardware/memory/cartridge/mapper_scabs.go
JetSetIlly da83fc311b removed complexity from cartridge fingerprinting process
all cartridge data is read through cartridgeloader io.Reader interface
2024-04-16 10:18:13 +01:00

208 lines
6.5 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 (
"fmt"
"io"
"github.com/jetsetilly/gopher2600/cartridgeloader"
"github.com/jetsetilly/gopher2600/environment"
"github.com/jetsetilly/gopher2600/hardware/memory/cartridge/mapper"
"github.com/jetsetilly/gopher2600/hardware/memory/memorymap"
)
// scabs implements the mapper.CartMapper interface.
//
// the function of the scheme is specified European Patent 84300730.3
// https://worldwide.espacenet.com/patent/search/family/023848640/publication/EP0116455A2?q=84300730.3
type scabs struct {
env *environment.Environment
mappingID string
bankSize int
banks [2][]uint8
state *scabsState
}
func newSCABS(env *environment.Environment, loader cartridgeloader.Loader) (mapper.CartMapper, error) {
data, err := io.ReadAll(loader)
if err != nil {
return nil, fmt.Errorf("FE: %w", err)
}
cart := &scabs{
env: env,
mappingID: "FE",
bankSize: 4096,
state: &scabsState{},
}
if len(data) != cart.bankSize*cart.NumBanks() {
return nil, fmt.Errorf("FE: wrong number of bytes in the cartridge data")
}
for k := 0; k < cart.NumBanks(); k++ {
cart.banks[k] = make([]uint8, cart.bankSize)
offset := k * cart.bankSize
copy(cart.banks[k], data[offset:offset+cart.bankSize])
}
return cart, nil
}
// MappedBanks implements the mapper.CartMapper interface.
func (cart *scabs) MappedBanks() string {
return fmt.Sprintf("Bank: %d", cart.state.bank)
}
// ID implements the mapper.CartMapper interface.
func (cart *scabs) ID() string {
return cart.mappingID
}
// Snapshot implements the mapper.CartMapper interface.
func (cart *scabs) Snapshot() mapper.CartMapper {
n := *cart
n.state = cart.state.Snapshot()
return &n
}
// Plumb implements the mapper.CartMapper interface.
func (cart *scabs) Plumb(env *environment.Environment) {
cart.env = env
}
// Reset implements the mapper.CartMapper interface.
func (cart *scabs) Reset() {
cart.state.bank = 0
}
// Access implements the mapper.CartMapper interface.
func (cart *scabs) Access(addr uint16, peek bool) (uint8, uint8, error) {
return cart.banks[cart.state.bank][addr], mapper.CartDrivenPins, nil
}
// AccessVolatile implements the mapper.CartMapper interface.
func (cart *scabs) AccessVolatile(_ uint16, _ uint8, _ bool) error {
return nil
}
// NumBanks implements the mapper.CartMapper interface.
func (cart *scabs) NumBanks() int {
return 2
}
// GetBank implements the mapper.CartMapper interface.
func (cart *scabs) GetBank(_ uint16) mapper.BankInfo {
return mapper.BankInfo{Number: cart.state.bank, IsRAM: false}
}
// AccessPassive implements the mapper.CartMapper interface.
func (cart *scabs) AccessPassive(addr uint16, data uint8) error {
// "[...] it will be noted that JSR instruction is always followed by an
// address 01FE on the address bus. One cycle thereafter the most
// significant 8bits of the new memory location appears on the data bus.
// Thus by monitoring the address bus for 01FE and then latching the most
// significant bit on the data bus cycle thereafter, memory bank selection
// can be implemented"
//
// Article 30 of European Patent 84300730.3
// the patent is very clear that the bank switch is performed one CPU cycle
// after 0x01fe appears on the data bus. it may seem odd therefore that we
// wait for two calls to AccessPassive() before completing the operation.
// however, the bankSwitch counter should not be thought of as a count of
// CPU cycles (but only as the number of calls to the function). the
// difference is caused by the ordering of the calls to AccessPassive()
//
// for "read" memory accesses the AccessPassive() is called in the moment
// between the address bus being set but the data bus not yet being set as a
// result of the access. from the point of view of the CPU we could think of
// these values as the "current" address bus and the "old" data bus
//
// the consequence of calling AccessPassive() in this ways means that we
// don't receive the critical data information until two calls after 0x01fe
// is seen on the address bus. this equates to one CPU cycle but the actual
// bank switch does not occur until the beginning of the cycle after the
// data bus is first set
// it is tempting to think that we can perform the bank switch as a result
// of the Access() itself. in other words, the Access() function calls a
// bankSwitch() function. however, this does not work because the bank can
// be switched without the data bus being set as a result of reading the
// cartridge. for example, the RTS function will read the data from RAM but
// still cause the cartridge to switch. this means that the bank switch can
// only occur as a consequence of AccessPassive()
switch cart.state.bankSwitch {
case 2:
cart.state.bankSwitch = 1
case 1:
switch data >> 5 {
case 0b111:
cart.state.bank = 0
case 0b110:
cart.state.bank = 1
}
cart.state.bankSwitch = 0
default:
if addr == 0x01fe {
cart.state.bankSwitch = 2
}
}
return nil
}
// Step implements the mapper.CartMapper interface.
func (cart *scabs) Step(_ float32) {
}
// IterateBank implements the mapper.CartMapper interface.
func (cart *scabs) CopyBanks() []mapper.BankContent {
c := make([]mapper.BankContent, len(cart.banks))
for b := 0; b < len(cart.banks); b++ {
c[b] = mapper.BankContent{Number: b,
Data: cart.banks[b],
Origins: []uint16{memorymap.OriginCart},
}
}
return c
}
// Patch implements the mapper.CartPatchable interface
func (cart *scabs) Patch(offset int, data uint8) error {
if offset >= cart.bankSize*len(cart.banks) {
return fmt.Errorf("FE: patch offset too high (%d)", offset)
}
bank := offset / cart.bankSize
offset %= cart.bankSize
cart.banks[bank][offset] = data
return nil
}
type scabsState struct {
bank int
bankSwitch int
}
// Snapshot implements the mapper.CartMapper interface.
func (s *scabsState) Snapshot() *scabsState {
n := *s
return &n
}