2020-10-03 23:38:26 +02:00
|
|
|
package izapple2
|
2019-02-24 15:05:50 +01:00
|
|
|
|
|
|
|
const (
|
|
|
|
ioDataKeyboard uint8 = 0x10
|
|
|
|
|
2019-04-26 18:08:30 +02:00
|
|
|
ioFlagText uint8 = 0x50
|
2019-02-24 15:05:50 +01:00
|
|
|
ioFlagMixed uint8 = 0x52
|
|
|
|
ioFlagSecondPage uint8 = 0x54
|
|
|
|
ioFlagHiRes uint8 = 0x56
|
2020-08-06 18:35:34 +02:00
|
|
|
ioFlagAnnunciator0 uint8 = 0x58
|
2019-02-24 15:05:50 +01:00
|
|
|
ioFlagAnnunciator1 uint8 = 0x5a
|
|
|
|
ioFlagAnnunciator2 uint8 = 0x5c
|
|
|
|
ioFlagAnnunciator3 uint8 = 0x5e
|
|
|
|
|
|
|
|
ioDataCassette uint8 = 0x60
|
|
|
|
ioFlagButton0 uint8 = 0x61
|
|
|
|
ioFlagButton1 uint8 = 0x62
|
|
|
|
ioFlagButton2 uint8 = 0x63
|
|
|
|
ioDataPaddle0 uint8 = 0x64
|
|
|
|
ioDataPaddle1 uint8 = 0x65
|
|
|
|
ioDataPaddle2 uint8 = 0x66
|
|
|
|
ioDataPaddle3 uint8 = 0x67
|
2020-08-06 18:35:34 +02:00
|
|
|
|
2020-09-23 18:12:28 +02:00
|
|
|
// Not real softSwitches. Using the numbers to store the flags somewhere.
|
|
|
|
ioFlagRGBCardActive uint8 = 0x7d
|
|
|
|
ioFlag1RGBCard uint8 = 0x7e
|
|
|
|
ioFlag2RGBCard uint8 = 0x7f
|
2019-02-24 15:05:50 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
func addApple2SoftSwitches(io *ioC0Page) {
|
|
|
|
|
2019-10-21 00:00:42 +02:00
|
|
|
io.addSoftSwitchRW(0x00, keySoftSwitch, "KEYBOARD") // Keyboard
|
|
|
|
io.addSoftSwitchRW(0x10, strobeKeyboardSoftSwitch, "AKD") // Keyboard Strobe
|
|
|
|
io.addSoftSwitchR(0x20, notImplementedSoftSwitchR, "TAPEOUT") // Cassette Output
|
|
|
|
io.addSoftSwitchR(0x30, speakerSoftSwitch, "SPEAKER") // Speaker
|
|
|
|
io.addSoftSwitchR(0x40, notImplementedSoftSwitchR, "STROBE") // Game connector Strobe
|
2019-02-24 15:05:50 +01:00
|
|
|
// Note: Some sources indicate that all these cover 16 positions
|
2020-08-13 16:43:40 +02:00
|
|
|
// for read and write. But the Apple2e takes over some of them, with
|
2019-02-24 15:05:50 +01:00
|
|
|
// the prevention on acting only on writes.
|
|
|
|
|
2019-10-21 00:00:42 +02:00
|
|
|
io.addSoftSwitchRW(0x50, getSoftSwitch(ioFlagText, false), "TEXTOFF")
|
|
|
|
io.addSoftSwitchRW(0x51, getSoftSwitch(ioFlagText, true), "TEXTON")
|
|
|
|
io.addSoftSwitchRW(0x52, getSoftSwitch(ioFlagMixed, false), "MIXEDOFF")
|
|
|
|
io.addSoftSwitchRW(0x53, getSoftSwitch(ioFlagMixed, true), "MIXEDON")
|
|
|
|
io.addSoftSwitchRW(0x54, getSoftSwitch(ioFlagSecondPage, false), "PAGE2OFF")
|
|
|
|
io.addSoftSwitchRW(0x55, getSoftSwitch(ioFlagSecondPage, true), "PAGE2ON")
|
|
|
|
io.addSoftSwitchRW(0x56, getSoftSwitch(ioFlagHiRes, false), "HIRESOFF")
|
|
|
|
io.addSoftSwitchRW(0x57, getSoftSwitch(ioFlagHiRes, true), "HIRESON")
|
2019-04-28 00:13:34 +02:00
|
|
|
|
2019-10-21 00:00:42 +02:00
|
|
|
io.addSoftSwitchRW(0x58, getSoftSwitch(ioFlagAnnunciator0, false), "ANN0OFF")
|
|
|
|
io.addSoftSwitchRW(0x59, getSoftSwitch(ioFlagAnnunciator0, true), "ANN0ON")
|
|
|
|
io.addSoftSwitchRW(0x5a, getSoftSwitch(ioFlagAnnunciator1, false), "ANN1OFF")
|
|
|
|
io.addSoftSwitchRW(0x5b, getSoftSwitch(ioFlagAnnunciator1, true), "ANN1ON")
|
|
|
|
io.addSoftSwitchRW(0x5c, getSoftSwitch(ioFlagAnnunciator2, false), "ANN2OFF")
|
|
|
|
io.addSoftSwitchRW(0x5d, getSoftSwitch(ioFlagAnnunciator2, true), "ANN2ON")
|
|
|
|
io.addSoftSwitchRW(0x5e, getSoftSwitch(ioFlagAnnunciator3, false), "ANN3OFF")
|
|
|
|
io.addSoftSwitchRW(0x5f, getSoftSwitch(ioFlagAnnunciator3, true), "ANN3ON")
|
2019-02-24 15:05:50 +01:00
|
|
|
|
2019-10-21 00:00:42 +02:00
|
|
|
io.addSoftSwitchR(0x60, notImplementedSoftSwitchR, "CASSETTE") // Cassette Input
|
|
|
|
io.addSoftSwitchR(0x61, getButtonSoftSwitch(0), "PB0")
|
|
|
|
io.addSoftSwitchR(0x62, getButtonSoftSwitch(1), "PB1")
|
|
|
|
io.addSoftSwitchR(0x63, getButtonSoftSwitch(2), "PB2")
|
|
|
|
io.addSoftSwitchR(0x64, getPaddleSoftSwitch(0), "PDL0")
|
|
|
|
io.addSoftSwitchR(0x65, getPaddleSoftSwitch(1), "PDL1")
|
|
|
|
io.addSoftSwitchR(0x66, getPaddleSoftSwitch(2), "PDL2")
|
|
|
|
io.addSoftSwitchR(0x67, getPaddleSoftSwitch(3), "PDL3")
|
2019-02-24 23:54:13 +01:00
|
|
|
|
|
|
|
// The previous 8 softswitches are repeated
|
2019-10-21 00:00:42 +02:00
|
|
|
io.addSoftSwitchR(0x68, notImplementedSoftSwitchR, "CASSETTE") // Cassette Input
|
|
|
|
io.addSoftSwitchR(0x69, getButtonSoftSwitch(0), "PB0")
|
|
|
|
io.addSoftSwitchR(0x6A, getButtonSoftSwitch(1), "PB1")
|
|
|
|
io.addSoftSwitchR(0x6B, getButtonSoftSwitch(2), "PB2")
|
|
|
|
io.addSoftSwitchR(0x6C, getPaddleSoftSwitch(0), "PDL0")
|
|
|
|
io.addSoftSwitchR(0x6D, getPaddleSoftSwitch(1), "PDL1")
|
|
|
|
io.addSoftSwitchR(0x6E, getPaddleSoftSwitch(2), "PDL2")
|
|
|
|
io.addSoftSwitchR(0x6F, getPaddleSoftSwitch(3), "PDL3")
|
2019-08-06 00:37:27 +02:00
|
|
|
|
2019-10-21 00:00:42 +02:00
|
|
|
io.addSoftSwitchR(0x70, strobePaddlesSoftSwitch, "RESETPDL") // Game controllers reset
|
2020-08-06 18:35:34 +02:00
|
|
|
|
|
|
|
// For RGB screen modes. Default to NTSC artifacts
|
|
|
|
io.softSwitchesData[ioFlag1RGBCard] = ssOn
|
|
|
|
io.softSwitchesData[ioFlag2RGBCard] = ssOn
|
2019-02-24 15:05:50 +01:00
|
|
|
}
|
|
|
|
|
2020-08-13 16:43:40 +02:00
|
|
|
func notImplementedSoftSwitchR(io *ioC0Page) uint8 {
|
|
|
|
// Return random info. Some games (Serpentine) used CASSETTE and get stuck if not changing.
|
|
|
|
return uint8(io.apple2.cpu.GetCycles())
|
2019-02-24 15:05:50 +01:00
|
|
|
}
|
2019-06-09 17:36:29 +02:00
|
|
|
|
|
|
|
func notImplementedSoftSwitchW(*ioC0Page, uint8) {
|
|
|
|
}
|
|
|
|
|
2019-11-11 22:58:42 +01:00
|
|
|
func setStatusSoftSwitch(ioFlag uint8) softSwitchW {
|
|
|
|
return func(io *ioC0Page, value uint8) {
|
|
|
|
io.softSwitchesData[ioFlag] = value
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-24 23:54:13 +01:00
|
|
|
func getStatusSoftSwitch(ioFlag uint8) softSwitchR {
|
|
|
|
return func(io *ioC0Page) uint8 {
|
2019-02-24 15:05:50 +01:00
|
|
|
return io.softSwitchesData[ioFlag]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-24 23:54:13 +01:00
|
|
|
func getSoftSwitch(ioFlag uint8, isSet bool) softSwitchR {
|
|
|
|
return func(io *ioC0Page) uint8 {
|
2019-02-24 15:05:50 +01:00
|
|
|
if isSet {
|
|
|
|
io.softSwitchesData[ioFlag] = ssOn
|
|
|
|
} else {
|
|
|
|
io.softSwitchesData[ioFlag] = ssOff
|
|
|
|
}
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-06 00:37:27 +02:00
|
|
|
func speakerSoftSwitch(io *ioC0Page) uint8 {
|
2019-05-10 00:09:15 +02:00
|
|
|
if io.speaker != nil {
|
|
|
|
io.speaker.Click(io.apple2.cpu.GetCycles())
|
|
|
|
}
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
2019-08-06 00:37:27 +02:00
|
|
|
func keySoftSwitch(io *ioC0Page) uint8 {
|
2019-04-14 20:46:40 +02:00
|
|
|
strobed := (io.softSwitchesData[ioDataKeyboard] & (1 << 7)) == 0
|
2019-04-14 12:54:11 +02:00
|
|
|
if io.keyboard != nil {
|
2019-04-14 20:46:40 +02:00
|
|
|
if key, ok := io.keyboard.GetKey(strobed); ok {
|
2019-02-24 23:54:13 +01:00
|
|
|
io.softSwitchesData[ioDataKeyboard] = key + (1 << 7)
|
2019-02-24 15:05:50 +01:00
|
|
|
}
|
|
|
|
}
|
2019-06-09 17:36:29 +02:00
|
|
|
value := io.softSwitchesData[ioDataKeyboard]
|
|
|
|
//fmt.Printf("Key $%02x, %v\n", value, strobed)
|
|
|
|
return value
|
|
|
|
|
2019-02-24 15:05:50 +01:00
|
|
|
}
|
|
|
|
|
2019-02-24 23:54:13 +01:00
|
|
|
func strobeKeyboardSoftSwitch(io *ioC0Page) uint8 {
|
|
|
|
result := io.softSwitchesData[ioDataKeyboard]
|
2019-06-09 17:36:29 +02:00
|
|
|
//fmt.Printf("Strobe $%02x\n", result)
|
2019-02-24 23:54:13 +01:00
|
|
|
io.softSwitchesData[ioDataKeyboard] &^= 1 << 7
|
2019-02-24 15:05:50 +01:00
|
|
|
return result
|
|
|
|
}
|
2019-08-06 00:37:27 +02:00
|
|
|
|
|
|
|
func getButtonSoftSwitch(i int) softSwitchR {
|
|
|
|
return func(io *ioC0Page) uint8 {
|
|
|
|
if io.joysticks != nil && io.joysticks.ReadButton(i) {
|
|
|
|
return 128
|
|
|
|
}
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Paddle values are calculated by the time taken by a current going
|
2020-08-06 18:35:34 +02:00
|
|
|
through the paddle variable resistor to charge a capacitor.
|
2019-08-06 00:37:27 +02:00
|
|
|
The capacitor is discharged via the strobe softswitch. The result is
|
2020-08-06 18:35:34 +02:00
|
|
|
how many times a 11 cycles loop runs before the capacitor reaches
|
2019-08-06 00:37:27 +02:00
|
|
|
the voltage threshold.
|
|
|
|
|
|
|
|
See: http://www.1000bit.it/support/manuali/apple/technotes/aiie/tn.aiie.06.html
|
|
|
|
*/
|
|
|
|
|
|
|
|
const paddleToCyclesFactor = 11
|
|
|
|
|
|
|
|
func getPaddleSoftSwitch(i int) softSwitchR {
|
|
|
|
return func(io *ioC0Page) uint8 {
|
|
|
|
if io.joysticks == nil {
|
2019-11-20 23:28:51 +01:00
|
|
|
return 255 // Capacitors never discharge if there is not joystick
|
|
|
|
}
|
|
|
|
reading, hasData := io.joysticks.ReadPaddle(i)
|
|
|
|
if !hasData {
|
|
|
|
return 255 // Capacitors never discharge if there is not joystick
|
2019-08-06 00:37:27 +02:00
|
|
|
}
|
|
|
|
cyclesNeeded := uint64(reading) * paddleToCyclesFactor
|
|
|
|
cyclesElapsed := io.apple2.cpu.GetCycles() - io.paddlesStrobeCycle
|
|
|
|
if cyclesElapsed < cyclesNeeded {
|
|
|
|
// The capacitor is not charged yet
|
|
|
|
return 128
|
|
|
|
}
|
2019-09-28 19:17:47 +02:00
|
|
|
return 0
|
2019-08-06 00:37:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func strobePaddlesSoftSwitch(io *ioC0Page) uint8 {
|
|
|
|
// On the real machine this discharges the capacitors.
|
|
|
|
io.paddlesStrobeCycle = io.apple2.cpu.GetCycles()
|
|
|
|
return 0
|
|
|
|
}
|