izapple2/cardDisk2Sequencer.go

277 lines
6.9 KiB
Go
Raw Normal View History

2021-05-09 17:48:54 +00:00
package izapple2
import (
"github.com/ivanizag/izapple2/component"
)
// TODO: fast mode
/*
See:
"Understanding the Apple II, chapter 9"
Beneath Apple ProDOS, Appendix
Phases: http://yesterbits.com/media/pubs/AppleOrchard/articles/disk-ii-part-1-1983-apr.pdf
IMW Floppy Disk I/O Controller info: (https://www.brutaldeluxe.fr/documentation/iwm/apple2_IWM_INFO_19840510.pdf)
Woz https://applesaucefdc.com/woz/reference2/
Schematic: https://mirrors.apple2.org.za/ftp.apple.asimov.net/documentation/hardware/schematics/APPLE_DiskII_SCH.pdf
*/
// CardDisk2Sequencer is a DiskII interface card with the Woz state machine
2021-05-09 17:48:54 +00:00
type CardDisk2Sequencer struct {
cardBase
p6ROM []uint8
2021-05-09 18:09:54 +00:00
q [8]bool // 8-bit latch SN74LS259
register uint8 // 8-bit shift/storage register SN74LS323
sequence uint8 // 4 bits stored in an hex flip-flop SN74LS174
motorDelay uint64 // NE556 timer, used to delay motor off
2021-05-09 17:48:54 +00:00
drive [2]cardDisk2SequencerDrive
2021-07-01 16:15:49 +00:00
lastWriteValue bool // We write transitions to the WOZ file. We store the last value to send a pulse on change.
2021-05-09 17:48:54 +00:00
lastPulseCycles uint8 // There is a new pulse every 4ms, that's 8 cycles of 2Mhz
lastCycle uint64 // 2 Mhz cycles
2022-02-25 23:05:09 +00:00
trackTracer trackTracer
2021-05-09 17:48:54 +00:00
}
2021-05-09 18:09:54 +00:00
const (
disk2MotorOffDelay = uint64(2 * 1000 * 1000) // 2 Mhz cycles. Total 1 second.
disk2PulseCyles = uint8(8) // 8 cycles = 4ms * 2Mhz
2021-05-09 18:09:54 +00:00
/*
We skip register calculations for long periods with the motor
on but not reading bytes. It's an optimizations, 10000 is too
short for cross track sync copy protections.
*/
disk2CyclestoLoseSsync = 100000
)
2021-05-09 17:48:54 +00:00
// NewCardDisk2Sequencer creates a new CardDisk2Sequencer
2022-02-25 23:05:09 +00:00
func NewCardDisk2Sequencer(trackTracer trackTracer) *CardDisk2Sequencer {
2021-05-09 17:48:54 +00:00
var c CardDisk2Sequencer
c.name = "Disk II"
2022-02-25 23:05:09 +00:00
c.trackTracer = trackTracer
2021-05-09 17:48:54 +00:00
c.loadRomFromResource("<internal>/DISK2.rom")
2022-01-28 18:25:52 +00:00
data, _, err := LoadResource("<internal>/DISK2P6.rom")
2021-05-09 17:48:54 +00:00
if err != nil {
// The resource should be internal and never fail
panic(err)
}
c.p6ROM = data
return &c
}
// GetInfo returns card info
func (c *CardDisk2Sequencer) GetInfo() map[string]string {
info := make(map[string]string)
info["rom"] = "16 sector"
// TODO: add drives info
return info
}
func (c *CardDisk2Sequencer) reset() {
// UtA2e 9-12, all switches forced to off
2021-05-09 18:09:54 +00:00
c.q = [8]bool{}
2021-05-09 17:48:54 +00:00
}
func (c *CardDisk2Sequencer) assign(a *Apple2, slot int) {
2022-08-05 17:43:17 +00:00
c.addCardSoftSwitches(func(address uint8, data uint8, write bool) uint8 {
2021-05-09 17:48:54 +00:00
/*
2021-05-09 18:09:54 +00:00
Slot card pins to SN74LS259 latch mapping:
2021-05-09 17:48:54 +00:00
slot_address[3,2,1] => latch_address[2,1,0]
slot_address[0] => latch_data
2021-05-09 18:09:54 +00:00
slot_dev_selct => latch_write_enable ;It will be true
2021-05-09 17:48:54 +00:00
*/
2021-05-09 18:09:54 +00:00
c.q[address>>1] = (address & 1) != 0
2021-05-09 17:48:54 +00:00
// Advance the Disk2 state machine since the last call to softswitches
c.catchUp(data)
2021-05-09 17:48:54 +00:00
/*
Slot card pins to SN74LS259 mapping:
slot_address[0] => latch_oe2_n
*/
register_output_enable_neg := (address & 1) != 0
if !register_output_enable_neg {
return c.register
} else {
2021-05-09 18:09:54 +00:00
return 33 // Floating
2021-05-09 17:48:54 +00:00
}
}, "DISK2SEQ")
c.cardBase.assign(a, slot)
}
func (c *CardDisk2Sequencer) catchUp(data uint8) {
currentCycle := c.a.cpu.GetCycles() << 1 // Disk2 cycles are x2 cpu cycle
motorOn := c.step(data, true)
2021-05-09 18:09:54 +00:00
if motorOn && currentCycle > c.lastCycle+disk2CyclestoLoseSsync {
2021-07-01 16:15:49 +00:00
// We have lost sync. We start the count.
// We do at least a couple 2 Mhz cycles
2021-05-09 17:48:54 +00:00
c.lastCycle = currentCycle - 2
}
c.lastCycle++
for motorOn && c.lastCycle <= currentCycle {
motorOn = c.step(data, false)
c.lastCycle++
}
if !motorOn {
2021-05-09 18:09:54 +00:00
c.lastCycle = 0 // Sync lost
2021-05-09 17:48:54 +00:00
}
}
func (c *CardDisk2Sequencer) step(data uint8, firstStep bool) bool {
/*
Q4 and Q6 set on the sofswitches is stored on the
latch.
*/
2021-05-09 18:09:54 +00:00
q5 := c.q[5] // Drive selection
q4 := c.q[4] // Motor on (before delay)
2021-05-09 17:48:54 +00:00
/*
Motor On comes from the latched q4 via the 556 to
provide a delay. The delay is reset while q4 is on.
*/
if q4 {
c.motorDelay = disk2MotorOffDelay
}
motorOn := c.motorDelay > 0
/*
The pins for the cable drives ENBL1 and ENBL2 are
connected to q5 and motor using half of the 74LS132
NAND to combine them.
*/
c.drive[0].enable(!q5 && motorOn)
c.drive[1].enable(q5 && motorOn)
/*
Motor on AND the 2 Mhz clock (Q3 pin 37 of the slot)
are connected to the clok pulse of the shift register
if off, the sequences does not advance. The and uses
another quarter of the 74LS132 NAND.
*/
if !motorOn {
c.sequence = 0
return false
}
c.motorDelay--
/*
Head movements. We assume it's instantaneous on Q0-Q3 change. We
will place it on the first step.
Q0 to Q3 are connected directly to the drives.
*/
if firstStep {
2021-05-09 18:09:54 +00:00
q0 := c.q[0]
q1 := c.q[1]
q2 := c.q[2]
q3 := c.q[3]
2022-02-25 23:05:09 +00:00
c.drive[0].moveHead(q0, q1, q2, q3, c.trackTracer)
c.drive[1].moveHead(q0, q1, q2, q3, c.trackTracer)
2021-05-09 17:48:54 +00:00
}
/*
The reading from the drive is converted to a pulse detecting
changes using Q3 and Q4 of the flip flop, combined with
the last quarter of the 74LS132 NAND.
The woz format provides the pulse directly and we won't emulate
this detection.
*/
pulse := false
c.lastPulseCycles++
if c.lastPulseCycles == disk2PulseCyles {
2021-07-01 16:15:49 +00:00
// Read
2021-05-09 17:48:54 +00:00
pulse = c.drive[0].readPulse() ||
c.drive[1].readPulse()
c.lastPulseCycles = 0
}
/*
The write protected signal comes directly from any of the
drives being enabled (motor on) and write protected.
*/
wProt := (c.drive[0].enabled && c.drive[0].writeProtected) ||
(c.drive[1].enabled && c.drive[1].writeProtected)
/*
The next instruction for the sequencer is retrieved from
the ROM P6 using the address:
A0, A5, A6, A7 <= sequence from 74LS174
A1 =< high, MSB of register (pin Q7)
A2 <= Q6 from 9334
A3 <= Q7 from 9334
A4 <= pulse transition
*/
high := c.register >= 0x80
seqBits := component.ByteToPins(c.sequence)
2021-05-09 17:48:54 +00:00
romAddress := component.PinsToByte([8]bool{
seqBits[1], // seq1
high,
2021-05-09 18:09:54 +00:00
c.q[6],
c.q[7],
2021-05-09 17:48:54 +00:00
!pulse,
seqBits[0], // seq0
seqBits[2], // seq2
2021-05-09 18:09:54 +00:00
seqBits[3], // seq3
2021-05-09 17:48:54 +00:00
})
romData := c.p6ROM[romAddress]
inst := romData & 0xf
next := romData >> 4
/*
The pins for the register shifter update are:
SR(CLR) <- ROM D3
S1 <- ROM D0
S0 <- ROM D1
DS0(SR) <- WPROT pin of the selected drive
DS7(SL) <- ROM D2
IO[7.0] <-> D[0-7] slot data bus (the order is reversed)
*/
if inst < 8 {
c.register = 0 // Bit 4 clear to reset
} else {
switch inst & 0x3 { // Bit 0 and 1 are the operation
case 0:
// Nothing
case 1:
// Shift left bringing bit 1
c.register = (c.register << 1) | ((inst >> 2) & 1)
case 2:
// Shift right bringing wProt
c.register = c.register >> 1
if wProt {
c.register |= 0x80
}
case 3:
// Load
c.register = data
}
2021-07-01 16:15:49 +00:00
if c.q[7] && (inst&0x3) != 0 {
currentWriteValue := next >= 0x8
writePulse := currentWriteValue != c.lastWriteValue
c.drive[0].writePulse(writePulse)
c.drive[1].writePulse(writePulse)
c.lastWriteValue = currentWriteValue
}
2021-05-09 17:48:54 +00:00
}
//fmt.Printf("[D2SEQ] Step. seq:%x inst:%x next:%x reg:%02x\n",
// c.sequence, inst, next, c.register)
c.sequence = next
2021-05-09 17:48:54 +00:00
return true
}