2020-10-03 23:38:26 +02:00
|
|
|
package izapple2
|
2019-05-17 23:28:20 +02:00
|
|
|
|
2019-10-06 01:26:00 +02:00
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
|
2022-01-28 18:09:59 +01:00
|
|
|
"github.com/ivanizag/iz6502"
|
2020-10-17 20:10:48 +02:00
|
|
|
"github.com/ivanizag/izapple2/storage"
|
2019-10-06 01:26:00 +02:00
|
|
|
)
|
2019-05-17 23:28:20 +02:00
|
|
|
|
2020-08-30 21:11:43 +02:00
|
|
|
func newApple2() *Apple2 {
|
2019-05-17 23:28:20 +02:00
|
|
|
var a Apple2
|
2019-10-12 21:37:37 +02:00
|
|
|
|
2020-08-30 21:11:43 +02:00
|
|
|
a.Name = "Pending"
|
2019-10-21 00:06:28 +02:00
|
|
|
a.mmu = newMemoryManager(&a)
|
2019-10-12 21:37:37 +02:00
|
|
|
a.io = newIoC0Page(&a)
|
|
|
|
return &a
|
|
|
|
}
|
|
|
|
|
2021-04-02 20:39:37 +02:00
|
|
|
func (a *Apple2) setup(clockMhz float64, fastMode bool) {
|
2019-05-17 23:28:20 +02:00
|
|
|
a.commandChannel = make(chan int, 100)
|
|
|
|
a.fastMode = fastMode
|
|
|
|
|
|
|
|
if clockMhz <= 0 {
|
|
|
|
// Full speed
|
|
|
|
a.cycleDurationNs = 0
|
|
|
|
} else {
|
|
|
|
a.cycleDurationNs = 1000.0 / clockMhz
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-30 21:11:43 +02:00
|
|
|
func setApple2plus(a *Apple2) {
|
|
|
|
a.Name = "Apple ][+"
|
2022-01-28 18:09:59 +01:00
|
|
|
a.cpu = iz6502.NewNMOS6502(a.mmu)
|
2020-08-30 21:11:43 +02:00
|
|
|
addApple2SoftSwitches(a.io)
|
|
|
|
}
|
|
|
|
|
|
|
|
func setApple2e(a *Apple2) {
|
|
|
|
a.Name = "Apple IIe"
|
|
|
|
a.isApple2e = true
|
2022-01-28 18:09:59 +01:00
|
|
|
a.cpu = iz6502.NewNMOS6502(a.mmu)
|
2020-08-30 21:11:43 +02:00
|
|
|
a.mmu.initExtendedRAM(1)
|
|
|
|
addApple2SoftSwitches(a.io)
|
|
|
|
addApple2ESoftSwitches(a.io)
|
|
|
|
}
|
|
|
|
|
|
|
|
func setApple2eEnhanced(a *Apple2) {
|
|
|
|
a.Name = "Apple //e"
|
|
|
|
a.isApple2e = true
|
2022-01-28 18:09:59 +01:00
|
|
|
a.cpu = iz6502.NewCMOS65c02(a.mmu)
|
2020-08-30 21:11:43 +02:00
|
|
|
a.mmu.initExtendedRAM(1)
|
|
|
|
addApple2SoftSwitches(a.io)
|
|
|
|
addApple2ESoftSwitches(a.io)
|
|
|
|
}
|
|
|
|
|
2021-04-02 20:39:37 +02:00
|
|
|
func (a *Apple2) addTracer(tracer executionTracer) {
|
|
|
|
if a.tracers == nil {
|
|
|
|
a.tracers = make([]executionTracer, 0)
|
|
|
|
}
|
|
|
|
|
|
|
|
a.tracers = append(a.tracers, tracer)
|
|
|
|
}
|
|
|
|
|
2020-10-14 00:26:47 +02:00
|
|
|
func (a *Apple2) insertCard(c Card, slot int) {
|
2019-05-18 16:43:51 +02:00
|
|
|
c.assign(a, slot)
|
|
|
|
a.cards[slot] = c
|
|
|
|
}
|
|
|
|
|
2020-10-14 00:26:47 +02:00
|
|
|
// GetCards returns the array of inserted cards
|
|
|
|
func (a *Apple2) GetCards() [8]Card {
|
|
|
|
return a.cards
|
|
|
|
}
|
|
|
|
|
2019-06-07 20:01:20 +02:00
|
|
|
const (
|
|
|
|
apple2RomSize = 12 * 1024
|
|
|
|
apple2eRomSize = 16 * 1024
|
|
|
|
)
|
|
|
|
|
|
|
|
// LoadRom loads a standard Apple2+ or 2e ROM
|
2019-10-06 01:26:00 +02:00
|
|
|
func (a *Apple2) LoadRom(filename string) error {
|
2020-11-03 10:51:12 -06:00
|
|
|
data, _, err := storage.LoadResource(filename)
|
2019-10-06 01:26:00 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-06-07 20:01:20 +02:00
|
|
|
size := len(data)
|
|
|
|
if size != apple2RomSize && size != apple2eRomSize {
|
2021-08-05 21:12:52 +02:00
|
|
|
return errors.New("rom size not supported")
|
2019-06-07 20:01:20 +02:00
|
|
|
}
|
|
|
|
|
2020-08-14 17:19:24 +02:00
|
|
|
romBase := 0x10000 - size
|
2020-09-23 18:19:15 +02:00
|
|
|
a.mmu.physicalROM[0] = newMemoryRangeROM(uint16(romBase), data, "Main ROM")
|
2019-10-06 01:26:00 +02:00
|
|
|
return nil
|
2019-06-07 20:01:20 +02:00
|
|
|
}
|
|
|
|
|
2019-10-02 23:39:39 +02:00
|
|
|
// AddDisk2 inserts a DiskII controller
|
2020-10-14 21:54:51 +02:00
|
|
|
func (a *Apple2) AddDisk2(slot int, diskImage, diskBImage string) error {
|
|
|
|
c := NewCardDisk2()
|
|
|
|
a.insertCard(c, slot)
|
2019-05-17 23:28:20 +02:00
|
|
|
|
|
|
|
if diskImage != "" {
|
2020-10-17 20:10:48 +02:00
|
|
|
diskette, err := storage.LoadDiskette(diskImage)
|
2019-10-06 01:26:00 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-10-14 21:54:51 +02:00
|
|
|
c.drive[0].insertDiskette(diskImage, diskette)
|
2019-05-17 23:28:20 +02:00
|
|
|
}
|
2019-10-06 01:26:00 +02:00
|
|
|
|
2020-04-12 18:29:04 -05:00
|
|
|
if diskBImage != "" {
|
2020-10-17 20:10:48 +02:00
|
|
|
diskette, err := storage.LoadDiskette(diskBImage)
|
2020-04-12 18:29:04 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-10-14 21:54:51 +02:00
|
|
|
c.drive[1].insertDiskette(diskImage, diskette)
|
2020-04-12 18:29:04 -05:00
|
|
|
}
|
|
|
|
|
2019-10-06 01:26:00 +02:00
|
|
|
return nil
|
2019-05-17 23:28:20 +02:00
|
|
|
}
|
|
|
|
|
2021-05-09 19:48:54 +02:00
|
|
|
// AddDisk2 inserts a DiskII controller
|
|
|
|
func (a *Apple2) AddDisk2Sequencer(slot int, diskImage, diskBImage string) error {
|
|
|
|
c := NewCardDisk2Sequencer()
|
|
|
|
a.insertCard(c, slot)
|
|
|
|
|
|
|
|
if diskImage != "" {
|
|
|
|
err := c.drive[0].insertDiskette(diskImage)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if diskBImage != "" {
|
|
|
|
err := c.drive[0].insertDiskette(diskBImage)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-08-11 23:53:05 +02:00
|
|
|
// AddSmartPortDisk adds a smart port card and image
|
|
|
|
func (a *Apple2) AddSmartPortDisk(slot int, hdImage string, trace bool) error {
|
2020-10-14 00:26:47 +02:00
|
|
|
c := NewCardHardDisk()
|
|
|
|
c.trace = trace
|
|
|
|
err := c.LoadImage(hdImage)
|
2019-10-06 01:26:00 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-10-14 00:26:47 +02:00
|
|
|
a.insertCard(c, slot)
|
2019-10-06 01:26:00 +02:00
|
|
|
return nil
|
2019-10-02 23:39:39 +02:00
|
|
|
}
|
|
|
|
|
2019-11-10 14:20:31 +01:00
|
|
|
// AddVidHD adds a card with the signature of VidHD
|
|
|
|
func (a *Apple2) AddVidHD(slot int) {
|
2020-10-14 21:54:51 +02:00
|
|
|
a.insertCard(NewCardVidHD(), slot)
|
2019-11-10 14:20:31 +01:00
|
|
|
}
|
|
|
|
|
2019-11-12 23:47:48 +01:00
|
|
|
// AddFastChip adds a card with the signature of VidHD
|
|
|
|
func (a *Apple2) AddFastChip(slot int) {
|
2020-10-14 21:54:51 +02:00
|
|
|
a.insertCard(NewCardFastChip(), slot)
|
2019-11-12 23:47:48 +01:00
|
|
|
}
|
|
|
|
|
2019-05-17 23:28:20 +02:00
|
|
|
// AddLanguageCard inserts a 16Kb card
|
|
|
|
func (a *Apple2) AddLanguageCard(slot int) {
|
2020-10-14 21:54:51 +02:00
|
|
|
a.insertCard(NewCardLanguage(), slot)
|
2019-05-17 23:28:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// AddSaturnCard inserts a 128Kb card
|
|
|
|
func (a *Apple2) AddSaturnCard(slot int) {
|
2020-10-14 21:54:51 +02:00
|
|
|
a.insertCard(NewCardSaturn(), slot)
|
2019-05-17 23:28:20 +02:00
|
|
|
}
|
|
|
|
|
2020-03-08 23:39:25 +01:00
|
|
|
// AddMemoryExpansionCard inserts an Apple II Memory Expansion card with 1GB
|
2020-10-14 21:54:51 +02:00
|
|
|
func (a *Apple2) AddMemoryExpansionCard(slot int) {
|
|
|
|
a.insertCard(NewCardMemoryExpansion(), slot)
|
2020-03-08 23:39:25 +01:00
|
|
|
}
|
|
|
|
|
2019-09-28 13:37:42 +02:00
|
|
|
// AddThunderClockPlusCard inserts a ThunderClock Plus clock card
|
2021-01-24 16:56:06 +01:00
|
|
|
func (a *Apple2) AddThunderClockPlusCard(slot int) {
|
|
|
|
a.insertCard(NewCardThunderClockPlus(), slot)
|
2019-09-28 13:37:42 +02:00
|
|
|
}
|
|
|
|
|
2021-01-24 23:25:52 +01:00
|
|
|
// AddMouseCard inserts a Mouse card
|
|
|
|
func (a *Apple2) AddMouseCard(slot int) {
|
|
|
|
a.insertCard(NewCardMouse(), slot)
|
|
|
|
}
|
|
|
|
|
2021-03-14 18:46:52 +01:00
|
|
|
// AddVidexCard inserts a Videx card
|
|
|
|
func (a *Apple2) AddVidexCard(slot int) {
|
|
|
|
c := NewCardVidex()
|
|
|
|
a.insertCard(c, slot)
|
|
|
|
a.softVideoSwitch = NewSoftVideoSwitch(c)
|
|
|
|
}
|
|
|
|
|
2020-08-06 18:35:34 +02:00
|
|
|
// AddRGBCard inserts an RBG option to the Apple IIe 80 col 64KB card
|
|
|
|
func (a *Apple2) AddRGBCard() {
|
|
|
|
setupRGBCard(a)
|
|
|
|
}
|
|
|
|
|
2020-08-11 23:53:05 +02:00
|
|
|
// AddRAMWorks inserts adds RAMWorks style RAM to the Apple IIe 80 col 64KB card
|
2020-08-10 20:52:17 +02:00
|
|
|
func (a *Apple2) AddRAMWorks(banks int) {
|
|
|
|
setupRAMWorksCard(a, banks)
|
|
|
|
}
|
|
|
|
|
2020-08-16 15:30:44 +02:00
|
|
|
// AddNoSlotClock inserts a DS1215 no slot clock under the main ROM
|
|
|
|
func (a *Apple2) AddNoSlotClock() {
|
|
|
|
nsc := newNoSlotClockDS1216(a, a.mmu.physicalROM[0])
|
|
|
|
a.mmu.physicalROM[0] = nsc
|
|
|
|
}
|
|
|
|
|
2020-10-25 00:22:52 +02:00
|
|
|
// AddRomX inserts a RomX. It intercepts all memory accesses
|
2021-10-12 12:26:40 +02:00
|
|
|
func (a *Apple2) AddRomX() error {
|
|
|
|
rx, err := newRomX(a, a.mmu)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-10-25 00:22:52 +02:00
|
|
|
a.cpu.SetMemory(rx)
|
2021-10-12 12:26:40 +02:00
|
|
|
return nil
|
2020-10-25 00:22:52 +02:00
|
|
|
}
|
|
|
|
|
2020-09-01 17:46:30 +02:00
|
|
|
// AddNoSlotClockInCard inserts a DS1215 no slot clock under a card ROM
|
|
|
|
func (a *Apple2) AddNoSlotClockInCard(slot int) error {
|
|
|
|
cardRom := a.mmu.cardsROM[slot]
|
|
|
|
if cardRom == nil {
|
2021-08-05 21:12:52 +02:00
|
|
|
return errors.New("no ROM available on the slot to add a no slot clock")
|
2020-09-01 17:46:30 +02:00
|
|
|
}
|
|
|
|
nsc := newNoSlotClockDS1216(a, cardRom)
|
|
|
|
a.mmu.cardsROM[slot] = nsc
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-05-18 23:40:59 +02:00
|
|
|
// AddCardLogger inserts a fake card that logs accesses
|
|
|
|
func (a *Apple2) AddCardLogger(slot int) {
|
2020-10-14 21:54:51 +02:00
|
|
|
c := NewCardLogger()
|
|
|
|
a.insertCard(c, slot)
|
2019-05-18 23:40:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// AddCardInOut inserts a fake card that interfaces with the emulator host
|
|
|
|
func (a *Apple2) AddCardInOut(slot int) {
|
2020-10-14 21:54:51 +02:00
|
|
|
c := NewCardInOut()
|
|
|
|
a.insertCard(c, slot)
|
2019-05-18 23:40:59 +02:00
|
|
|
}
|
|
|
|
|
2019-05-17 23:28:20 +02: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-06 00:37:27 +02:00
|
|
|
|
|
|
|
// SetJoysticksProvider attaches an external joysticks provider
|
|
|
|
func (a *Apple2) SetJoysticksProvider(j JoysticksProvider) {
|
|
|
|
a.io.setJoysticksProvider(j)
|
|
|
|
}
|
2021-01-24 23:25:52 +01:00
|
|
|
|
|
|
|
// SetMouseProvider attaches an external joysticks provider
|
|
|
|
func (a *Apple2) SetMouseProvider(m MouseProvider) {
|
|
|
|
a.io.setMouseProvider(m)
|
|
|
|
}
|