izapple2/cardInOut.go

156 lines
2.8 KiB
Go
Raw Normal View History

2020-10-03 23:38:26 +02:00
package izapple2
2019-05-18 23:40:59 +02:00
import (
"fmt"
)
/*
In out card experiment to interface with the emulator host.
See:
"Apple II Monitors peeled."
http://mysite.du.edu/~etuttle/math/acia.htm
*/
2020-10-14 21:54:51 +02:00
// CardInOut is an experimental card to bridge with the host
type CardInOut struct {
2019-05-18 23:40:59 +02:00
cardBase
i int
}
2020-10-14 21:54:51 +02:00
// NewCardInOut creates CardInOut
func NewCardInOut() *CardInOut {
var c CardInOut
c.name = "Card to test I/O"
return &c
}
func (c *CardInOut) assign(a *Apple2, slot int) {
2019-10-21 00:00:42 +02:00
for i := uint8(0x0); i <= 0xf; i++ {
2019-05-18 23:40:59 +02:00
iCopy := i
2019-10-21 00:00:42 +02:00
c.addCardSoftSwitchR(i, func(*ioC0Page) uint8 {
2019-05-18 23:40:59 +02:00
value := []uint8{0xc1, 0xc1, 0x93, 0x0}[c.i%4]
c.i++
fmt.Printf("[cardInOut] Read access to softswith 0x%x for slot %v, value %x.\n", iCopy, slot, value)
//return 0x41 + 0x80
return []uint8{0x41, 0x41, 0x13}[i%3] + 0x80
2019-10-21 00:00:42 +02:00
}, "INOUTR")
c.addCardSoftSwitchW(i, func(_ *ioC0Page, value uint8) {
2019-05-18 23:40:59 +02:00
fmt.Printf("[cardInOut] Write access to softswith 0x%x for slot %v, value 0x%x.\n", iCopy, slot, value)
2019-10-21 00:00:42 +02:00
}, "INOUTW")
2019-05-18 23:40:59 +02:00
}
in := true
out := false
data := [256]uint8{
// Register
0xA9, 0xC2,
0x85, 0x37,
0x85, 0x39,
0xA9, 0x10,
0x85, 0x36,
0xA9, 0x15,
0x85, 0x38,
0x60, 0xEA,
// Out char
0x8D, 0xA1, 0xC0,
0x60, 0xEA,
// Get char
0x91, 0x28,
0xAD, 0xA0, 0xC0,
0x60,
}
if !out {
// NOP the CSWL,H change
for _, v := range []uint8{2, 3, 8, 9} {
data[v] = 0xEA
}
}
if !in {
// NOP the KSWL,H change
for _, v := range []uint8{4, 5, 12, 13} {
data[v] = 0xEA
}
}
2020-09-23 18:19:15 +02:00
c.romCsxx = newMemoryRangeROM(0xC200, data[0:255], "InOUt card")
2019-05-18 23:40:59 +02:00
if slot != 2 {
2020-09-23 18:19:15 +02:00
// To make it work on other slots, patch C2, A0 and A1
2019-10-06 01:26:00 +02:00
panic("Assert failed. Only slot 2 supported for the InOut card")
2019-05-18 23:40:59 +02:00
}
c.cardBase.assign(a, slot)
}
/*
The ROM code was assembled using https://www.masswerk.at/6502/assembler.html
src:
BASL = $28
CSWL = $36
CSWH = $37
KSWL = $38
KSWH = $39
* = $C200
Register:
LDA #$C2
STA *CSWH
STA *KSWH
LDA #$10
STA *CSWL
LDA #$15
STA *KSWL
RTS
NOP
OutChar:
STA $C0A1
RTS
NOP
GetChar:
STA (BASL),Y
LDA $C0A0
RTS
assembled as:
0000 BASL = 0028
0000 CSWL = 0036
0000 CSWH = 0037
0000 KSWL = 0038
0000 KSWH = 0039
* = $C200
C200 REGIST
C200 LDA #$C2 A9 C2
C202 STA *CSWH 85 37
C204 STA *KSWH 85 39
C206 LDA #$10 A9 10
C208 STA *CSWL 85 36
C20A LDA #$15 A9 15
C20C STA *KSWL 85 38
C20E RTS 60
C20F NOP EA
C210 OUTCHA
C210 STA $C0A1 8D A1 C0
C213 RTS 60
C214 NOP EA
C215 GETCHA
C215 STA (BASL),Y 91 28
C217 LDA $C0A0 AD A0 C0
C21A RTS
object code:
A9 C2 85 37 85 39 A9 10
85 36 A9 15 85 38 60 EA
8D A1 C0 60 EA 91 28 AD
A0 C0 60
*/