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
|
|
|
|
|
|
|
|
*/
|