2019-05-17 21:28:20 +00:00
|
|
|
package apple2
|
|
|
|
|
2019-10-05 23:26:00 +00:00
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
|
|
|
|
"github.com/ivanizag/apple2/core6502"
|
|
|
|
)
|
2019-05-17 21:28:20 +00:00
|
|
|
|
2019-10-12 19:37:37 +00:00
|
|
|
// newApple2 instantiates an apple2
|
|
|
|
func newApple2plus() *Apple2 {
|
2019-05-17 21:28:20 +00:00
|
|
|
var a Apple2
|
2019-06-09 21:54:27 +00:00
|
|
|
a.Name = "Apple ][+"
|
2019-06-07 18:01:20 +00:00
|
|
|
a.mmu = newMemoryManager(&a)
|
2019-05-17 21:28:20 +00:00
|
|
|
a.cpu = core6502.NewNMOS6502(a.mmu)
|
2019-10-12 19:37:37 +00:00
|
|
|
a.io = newIoC0Page(&a)
|
|
|
|
addApple2SoftSwitches(a.io)
|
|
|
|
|
|
|
|
return &a
|
|
|
|
}
|
|
|
|
|
2019-10-20 22:06:28 +00:00
|
|
|
func newApple2e() *Apple2 {
|
|
|
|
var a Apple2
|
|
|
|
a.Name = "Apple IIe"
|
|
|
|
a.mmu = newMemoryManager(&a)
|
2019-11-05 23:32:34 +00:00
|
|
|
a.cpu = core6502.NewNMOS6502(a.mmu)
|
2019-10-20 22:06:28 +00:00
|
|
|
a.io = newIoC0Page(&a)
|
2019-11-03 23:23:03 +00:00
|
|
|
a.mmu.InitRAMalt()
|
2019-10-20 22:06:28 +00:00
|
|
|
addApple2SoftSwitches(a.io)
|
|
|
|
addApple2ESoftSwitches(a.io)
|
|
|
|
|
|
|
|
return &a
|
|
|
|
}
|
|
|
|
|
2019-10-12 19:37:37 +00:00
|
|
|
func newApple2eEnhanced() *Apple2 {
|
|
|
|
var a Apple2
|
|
|
|
a.Name = "Apple //e"
|
|
|
|
a.mmu = newMemoryManager(&a)
|
|
|
|
a.cpu = core6502.NewCMOS65c02(a.mmu)
|
|
|
|
a.io = newIoC0Page(&a)
|
2019-11-03 23:23:03 +00:00
|
|
|
a.mmu.InitRAMalt()
|
2019-10-12 19:37:37 +00:00
|
|
|
addApple2SoftSwitches(a.io)
|
|
|
|
addApple2ESoftSwitches(a.io)
|
|
|
|
|
|
|
|
return &a
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *Apple2) setup(isColor bool, clockMhz float64, fastMode bool) {
|
2019-05-17 21:28:20 +00:00
|
|
|
a.commandChannel = make(chan int, 100)
|
|
|
|
a.isColor = isColor
|
|
|
|
a.fastMode = fastMode
|
|
|
|
|
|
|
|
if clockMhz <= 0 {
|
|
|
|
// Full speed
|
|
|
|
a.cycleDurationNs = 0
|
|
|
|
} else {
|
|
|
|
a.cycleDurationNs = 1000.0 / clockMhz
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-18 14:43:51 +00:00
|
|
|
func (a *Apple2) insertCard(c card, slot int) {
|
|
|
|
c.assign(a, slot)
|
|
|
|
a.cards[slot] = c
|
|
|
|
}
|
|
|
|
|
2019-06-07 18:01:20 +00:00
|
|
|
const (
|
|
|
|
apple2RomSize = 12 * 1024
|
|
|
|
apple2eRomSize = 16 * 1024
|
|
|
|
)
|
|
|
|
|
|
|
|
// LoadRom loads a standard Apple2+ or 2e ROM
|
2019-10-05 23:26:00 +00:00
|
|
|
func (a *Apple2) LoadRom(filename string) error {
|
|
|
|
data, err := loadResource(filename)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-06-07 18:01:20 +00:00
|
|
|
size := len(data)
|
|
|
|
if size != apple2RomSize && size != apple2eRomSize {
|
2019-10-05 23:26:00 +00:00
|
|
|
return errors.New("Rom size not supported")
|
2019-06-07 18:01:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
romStart := 0
|
|
|
|
mmu := a.mmu
|
|
|
|
if size == apple2eRomSize {
|
|
|
|
// The extra 4kb ROM is first in the rom file.
|
|
|
|
// It starts with 256 unused bytes not mapped to 0xc000.
|
|
|
|
a.isApple2e = true
|
|
|
|
extraRomSize := apple2eRomSize - apple2RomSize
|
|
|
|
mmu.physicalROMe = newMemoryRange(0xc000, data[0:extraRomSize])
|
|
|
|
romStart = extraRomSize
|
|
|
|
}
|
|
|
|
|
2019-11-03 17:22:10 +00:00
|
|
|
mmu.physicalROM[0] = newMemoryRange(0xd000, data[romStart:])
|
2019-10-05 23:26:00 +00:00
|
|
|
return nil
|
2019-06-07 18:01:20 +00:00
|
|
|
}
|
|
|
|
|
2019-10-02 21:39:39 +00:00
|
|
|
// AddDisk2 inserts a DiskII controller
|
2019-10-05 23:26:00 +00:00
|
|
|
func (a *Apple2) AddDisk2(slot int, diskRomFile string, diskImage string) error {
|
2019-05-18 14:43:51 +00:00
|
|
|
var c cardDisk2
|
2019-10-05 23:26:00 +00:00
|
|
|
data, err := loadResource(diskRomFile)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
c.loadRom(data)
|
2019-05-18 14:43:51 +00:00
|
|
|
a.insertCard(&c, slot)
|
2019-05-17 21:28:20 +00:00
|
|
|
|
|
|
|
if diskImage != "" {
|
2019-12-22 13:15:19 +00:00
|
|
|
diskette, err := loadDisquette(diskImage)
|
2019-10-05 23:26:00 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-05-18 14:43:51 +00:00
|
|
|
c.drive[0].insertDiskette(diskette)
|
2019-05-17 21:28:20 +00:00
|
|
|
}
|
2019-10-05 23:26:00 +00:00
|
|
|
|
|
|
|
return nil
|
2019-05-17 21:28:20 +00:00
|
|
|
}
|
|
|
|
|
2019-10-02 21:39:39 +00:00
|
|
|
// AddHardDisk adds a ProDos hard dirve with a 2MG image
|
2019-11-01 17:48:39 +00:00
|
|
|
func (a *Apple2) AddHardDisk(slot int, hdImage string, trace bool) error {
|
2019-10-02 21:39:39 +00:00
|
|
|
var c cardHardDisk
|
2019-11-01 17:48:39 +00:00
|
|
|
c.setTrace(trace)
|
2019-10-02 21:39:39 +00:00
|
|
|
c.loadRom(buildHardDiskRom(slot))
|
|
|
|
a.insertCard(&c, slot)
|
|
|
|
|
2019-10-05 23:26:00 +00:00
|
|
|
hd, err := openHardDisk2mg(hdImage)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-10-02 21:39:39 +00:00
|
|
|
c.addDisk(hd)
|
2019-10-05 23:26:00 +00:00
|
|
|
return nil
|
2019-10-02 21:39:39 +00:00
|
|
|
}
|
|
|
|
|
2019-11-10 13:20:31 +00:00
|
|
|
// AddVidHD adds a card with the signature of VidHD
|
|
|
|
func (a *Apple2) AddVidHD(slot int) {
|
|
|
|
var c cardVidHD
|
|
|
|
c.loadRom(buildVidHDRom())
|
|
|
|
a.insertCard(&c, slot)
|
|
|
|
}
|
|
|
|
|
2019-11-12 22:47:48 +00:00
|
|
|
// AddFastChip adds a card with the signature of VidHD
|
|
|
|
func (a *Apple2) AddFastChip(slot int) {
|
|
|
|
var c cardFastChip
|
|
|
|
c.loadRom(buildFastChipRom())
|
|
|
|
a.insertCard(&c, slot)
|
|
|
|
}
|
|
|
|
|
2019-05-17 21:28:20 +00:00
|
|
|
// AddLanguageCard inserts a 16Kb card
|
|
|
|
func (a *Apple2) AddLanguageCard(slot int) {
|
2019-05-18 14:43:51 +00:00
|
|
|
a.insertCard(&cardLanguage{}, slot)
|
2019-05-17 21:28:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// AddSaturnCard inserts a 128Kb card
|
|
|
|
func (a *Apple2) AddSaturnCard(slot int) {
|
2019-05-18 14:43:51 +00:00
|
|
|
a.insertCard(&cardSaturn{}, slot)
|
2019-05-17 21:28:20 +00:00
|
|
|
}
|
|
|
|
|
2019-09-28 11:37:42 +00:00
|
|
|
// AddThunderClockPlusCard inserts a ThunderClock Plus clock card
|
2019-10-05 23:26:00 +00:00
|
|
|
func (a *Apple2) AddThunderClockPlusCard(slot int, romFile string) error {
|
2019-09-28 11:37:42 +00:00
|
|
|
var c cardThunderClockPlus
|
2019-10-05 23:26:00 +00:00
|
|
|
data, err := loadResource(romFile)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
c.loadRom(data)
|
2019-09-28 11:37:42 +00:00
|
|
|
a.insertCard(&c, slot)
|
2019-10-05 23:26:00 +00:00
|
|
|
return nil
|
2019-09-28 11:37:42 +00:00
|
|
|
}
|
|
|
|
|
2019-05-18 21:40:59 +00:00
|
|
|
// AddCardLogger inserts a fake card that logs accesses
|
|
|
|
func (a *Apple2) AddCardLogger(slot int) {
|
|
|
|
a.insertCard(&cardLogger{}, slot)
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddCardInOut inserts a fake card that interfaces with the emulator host
|
|
|
|
func (a *Apple2) AddCardInOut(slot int) {
|
|
|
|
a.insertCard(&cardInOut{}, slot)
|
|
|
|
}
|
|
|
|
|
2019-05-17 21:28:20 +00:00
|
|
|
// SetKeyboardProvider attaches an external keyboard provider
|
|
|
|
func (a *Apple2) SetKeyboardProvider(kb KeyboardProvider) {
|
|
|
|
a.io.setKeyboardProvider(kb)
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetSpeakerProvider attaches an external keyboard provider
|
|
|
|
func (a *Apple2) SetSpeakerProvider(s SpeakerProvider) {
|
|
|
|
a.io.setSpeakerProvider(s)
|
|
|
|
}
|
2019-08-05 22:37:27 +00:00
|
|
|
|
|
|
|
// SetJoysticksProvider attaches an external joysticks provider
|
|
|
|
func (a *Apple2) SetJoysticksProvider(j JoysticksProvider) {
|
|
|
|
a.io.setJoysticksProvider(j)
|
|
|
|
}
|