2019-02-16 19:15:41 +00:00
|
|
|
package apple2
|
2019-02-16 16:32:06 +00:00
|
|
|
|
2019-02-17 23:01:48 +00:00
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
)
|
2019-02-16 16:32:06 +00:00
|
|
|
|
|
|
|
type ioC0Page struct {
|
2019-03-02 19:41:25 +00:00
|
|
|
softSwitchesR [256]softSwitchR
|
|
|
|
softSwitchesW [256]softSwitchW
|
2019-02-23 23:41:32 +00:00
|
|
|
softSwitchesData [128]uint8
|
|
|
|
keyboard keyboardProvider
|
2019-03-02 17:33:50 +00:00
|
|
|
apple2 *Apple2
|
2019-02-20 22:51:47 +00:00
|
|
|
}
|
|
|
|
|
2019-02-24 22:54:13 +00:00
|
|
|
type softSwitchR func(io *ioC0Page) uint8
|
|
|
|
type softSwitchW func(io *ioC0Page, value uint8)
|
2019-02-23 23:41:32 +00:00
|
|
|
|
2019-02-20 22:51:47 +00:00
|
|
|
type keyboardProvider interface {
|
|
|
|
getKey() (key uint8, ok bool)
|
2019-02-16 16:32:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// See https://www.kreativekorp.com/miscpages/a2info/iomemory.shtml
|
|
|
|
// See https://stason.org/TULARC/pc/apple2/programmer/004-I-d-like-to-do-some-serious-Apple-II-programming-Whe.html
|
|
|
|
|
|
|
|
const (
|
2019-02-24 13:37:10 +00:00
|
|
|
ssOn uint8 = 0x80
|
|
|
|
ssOff uint8 = 0x00
|
|
|
|
)
|
|
|
|
|
2019-03-02 17:33:50 +00:00
|
|
|
func newIoC0Page(a *Apple2) *ioC0Page {
|
2019-02-24 14:05:50 +00:00
|
|
|
var io ioC0Page
|
2019-03-02 17:33:50 +00:00
|
|
|
io.apple2 = a
|
2019-02-16 16:32:06 +00:00
|
|
|
|
2019-02-24 14:05:50 +00:00
|
|
|
addApple2SoftSwitches(&io)
|
2019-03-02 17:33:50 +00:00
|
|
|
if a.isApple2e {
|
2019-02-24 14:05:50 +00:00
|
|
|
addApple2ESoftSwitches(&io)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &io
|
2019-02-23 23:41:32 +00:00
|
|
|
}
|
|
|
|
|
2019-02-24 22:54:13 +00:00
|
|
|
func (p *ioC0Page) addSoftSwitchRW(address uint8, ss softSwitchR) {
|
|
|
|
p.addSoftSwitchR(address, ss)
|
|
|
|
p.addSoftSwitchW(address, func(p *ioC0Page, _ uint8) {
|
|
|
|
ss(p)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *ioC0Page) addSoftSwitchR(address uint8, ss softSwitchR) {
|
|
|
|
if p.softSwitchesR[address] != nil {
|
|
|
|
fmt.Printf("Addresss 0x0c%02x is already assigned for read", address)
|
|
|
|
}
|
|
|
|
p.softSwitchesR[address] = ss
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *ioC0Page) addSoftSwitchW(address uint8, ss softSwitchW) {
|
|
|
|
if p.softSwitchesW[address] != nil {
|
|
|
|
fmt.Printf("Addresss 0x0c%02x is already assigned for write", address)
|
|
|
|
}
|
|
|
|
p.softSwitchesW[address] = ss
|
|
|
|
}
|
|
|
|
|
2019-02-24 14:05:50 +00:00
|
|
|
func (p *ioC0Page) isSoftSwitchExtActive(ioFlag uint8) bool {
|
|
|
|
return (p.softSwitchesData[ioFlag] & ssOn) == ssOn
|
2019-02-16 16:32:06 +00:00
|
|
|
}
|
|
|
|
|
2019-02-23 23:41:32 +00:00
|
|
|
func (p *ioC0Page) setKeyboardProvider(kb keyboardProvider) {
|
|
|
|
p.keyboard = kb
|
2019-02-16 16:32:06 +00:00
|
|
|
}
|
|
|
|
|
2019-02-16 19:15:41 +00:00
|
|
|
func (p *ioC0Page) Peek(address uint8) uint8 {
|
2019-02-16 16:32:06 +00:00
|
|
|
//fmt.Printf("Peek on $C0%02x ", address)
|
2019-02-24 22:54:13 +00:00
|
|
|
ss := p.softSwitchesR[address]
|
|
|
|
if ss == nil {
|
2019-03-04 23:00:12 +00:00
|
|
|
//panic(fmt.Sprintf("Unknown softswitch on read to 0xC0%02x", address))
|
|
|
|
return 0
|
2019-02-24 22:54:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return ss(p)
|
2019-02-16 16:32:06 +00:00
|
|
|
}
|
|
|
|
|
2019-02-16 19:15:41 +00:00
|
|
|
func (p *ioC0Page) Poke(address uint8, value uint8) {
|
2019-02-16 16:32:06 +00:00
|
|
|
//fmt.Printf("Poke on $C0%02x with %02x ", address, value)
|
2019-02-24 22:54:13 +00:00
|
|
|
ss := p.softSwitchesW[address]
|
2019-02-23 23:41:32 +00:00
|
|
|
if ss == nil {
|
2019-03-04 23:00:12 +00:00
|
|
|
//panic(fmt.Sprintf("Unknown softswitch on write to 0xC0%02x", address))
|
|
|
|
return
|
2019-02-23 23:41:32 +00:00
|
|
|
}
|
2019-02-24 22:54:13 +00:00
|
|
|
ss(p, value)
|
2019-02-23 23:41:32 +00:00
|
|
|
}
|