izapple2/cardLanguage.go

126 lines
2.9 KiB
Go
Raw Normal View History

2020-10-03 23:38:26 +02:00
package izapple2
2019-05-16 22:55:19 +02:00
/*
Language card with 16 extra kb for the Apple ][ and ][+
Manual: http://www.applelogic.org/files/LANGCARDMAN.pdf
The language card doesn't have ROM for Cx00. It would not
be used in slot 0 anyway.
Note also that language cards for the Apple ][ had ROM on
board to replace the main board F8 ROM with Autostart. That
was not used/needed on the Apple ][+. As this emulates the
Apple ][+, it is not considered. For the Plus it is often
2020-10-07 09:37:13 +02:00
referred as Language card but it is really a 16 KB Ram card,
2019-05-16 22:55:19 +02:00
"When RAM is deselected, the ROM on the Language card is selected for
2019-05-16 22:55:19 +02:00
the top 2K ($F800-$FFFF), and the ROM on the main board is selected
for $D000-$F7FF.
Power on RESET initializes ROM to read mode and RAM to write mode,
and selects the second 4K bank to map $D000-$DFFF."
2020-10-14 21:54:51 +02:00
Writing to the softswitch disables writing in LC? Saw that
somewhere but doing so fails IIe self check.
2019-05-16 22:55:19 +02:00
*/
2020-10-14 21:54:51 +02:00
// CardLanguage is an Language Card
type CardLanguage struct {
2019-05-16 22:55:19 +02:00
cardBase
readState bool
2019-10-21 00:00:42 +02:00
writeState uint8
2019-11-06 00:02:03 +01:00
altBank bool // false is bank1, true is bank2
2019-05-16 22:55:19 +02:00
}
2020-10-14 21:54:51 +02:00
// NewCardLanguage creates a new CardLanguage
func NewCardLanguage() *CardLanguage {
var c CardLanguage
c.name = "16KB Language Card"
return &c
}
2019-05-16 22:55:19 +02:00
const (
// Write enabling requires two softswitch accesses
2019-05-16 22:55:19 +02:00
lcWriteDisabled = 0
lcWriteHalfEnabled = 1
lcWriteEnabled = 2
)
2020-10-28 00:43:33 +01:00
func (c *CardLanguage) reset() {
if c.a.isApple2e {
// UtA2e 1-3, 5-23
c.readState = false
c.writeState = lcWriteEnabled
c.altBank = true // Start on bank2
c.applyState()
}
}
2020-10-14 21:54:51 +02:00
func (c *CardLanguage) assign(a *Apple2, slot int) {
2019-05-16 22:55:19 +02:00
c.readState = false
c.writeState = lcWriteEnabled
2019-11-06 00:02:03 +01:00
c.altBank = true // Start on bank2
2019-05-16 22:55:19 +02:00
a.mmu.initLanguageRAM(1)
2019-10-21 00:00:42 +02:00
for i := uint8(0x0); i <= 0xf; i++ {
2019-05-16 22:55:19 +02:00
iCopy := i
2019-10-21 00:00:42 +02:00
c.addCardSoftSwitchR(iCopy, func(*ioC0Page) uint8 {
c.ssAction(iCopy, false)
2019-05-16 22:55:19 +02:00
return 0
2019-10-21 00:00:42 +02:00
}, "LANGCARDR")
c.addCardSoftSwitchW(iCopy, func(*ioC0Page, uint8) {
c.ssAction(iCopy, true)
2019-10-21 00:00:42 +02:00
}, "LANGCARDW")
2019-05-16 22:55:19 +02:00
}
2019-05-18 16:43:51 +02:00
c.cardBase.assign(a, slot)
c.applyState()
2019-05-16 22:55:19 +02:00
}
2020-10-14 21:54:51 +02:00
func (c *CardLanguage) ssAction(ss uint8, write bool) {
2019-11-06 00:02:03 +01:00
c.altBank = ((ss >> 3) & 1) == 0
2019-05-16 22:55:19 +02:00
action := ss & 0x3
switch action {
case 0:
// RAM read, no writes
c.readState = true
c.writeState = lcWriteDisabled
case 1:
// ROM read, RAM write
c.readState = false
if !write {
c.writeState++
}
2019-05-16 22:55:19 +02:00
case 2:
// ROM read, no writes
c.readState = false
c.writeState = lcWriteDisabled
case 3:
//RAM read, RAM write
c.readState = true
if !write {
c.writeState++
}
}
if write && c.writeState == lcWriteHalfEnabled {
// UtA2e, 5-23. It is reset by even read access or any write acccess in the $C08x range
// And https://github.com/zellyn/a2audit/issues/3
c.writeState = lcWriteDisabled
2019-05-16 22:55:19 +02:00
}
if c.writeState > lcWriteEnabled {
c.writeState = lcWriteEnabled
}
c.applyState()
}
2020-10-14 21:54:51 +02:00
func (c *CardLanguage) applyState() {
c.a.mmu.setLanguageRAM(c.readState, c.writeState == lcWriteEnabled, c.altBank)
2019-05-16 22:55:19 +02:00
}