izapple2/cardDisk2.go

276 lines
7.2 KiB
Go
Raw Normal View History

2020-10-03 23:38:26 +02:00
package izapple2
import (
2019-10-21 00:00:42 +02:00
"fmt"
2020-10-14 21:54:51 +02:00
"strconv"
)
2019-03-02 20:41:25 +01:00
/*
https://applesaucefdc.com/woz/reference2/
2019-03-02 20:41:25 +01:00
Good explanation of the softswitches and the phases:
http://yesterbits.com/media/pubs/AppleOrchard/articles/disk-ii-part-1-1983-apr.pdf
2019-03-05 00:00:12 +01:00
"IMW Floppy Disk I/O Controller info" (https://www.brutaldeluxe.fr/documentation/iwm/apple2_IWM_INFO_19840510.pdf)
"Understanfing the Apple II, chapter 9"
2019-03-05 00:00:12 +01:00
35 tracks, 16 sectors, 256 bytes
NIB: 35 tracks 6656 bytes, 232960 bytes
2019-03-02 20:41:25 +01:00
*/
2020-10-16 17:13:14 +02:00
// CardDisk2 is a DiskII interface card
2020-10-14 21:54:51 +02:00
type CardDisk2 struct {
2019-03-02 20:41:25 +01:00
cardBase
selected int // q5, Only 0 and 1 supported
2019-03-05 00:00:12 +01:00
drive [2]cardDisk2Drive
dataLatch uint8
q6 bool
q7 bool
2019-03-05 00:00:12 +01:00
}
type cardDisk2Drive struct {
2020-10-14 21:54:51 +02:00
name string
diskette diskette
power bool // q4
phases uint8 // q3, q2, q1 and q0 with q0 on the LSB. Magnets that are active on the stepper motor
tracksStep int // Stepmotor for tracks position. 4 steps per track
}
const (
2020-10-14 21:54:51 +02:00
maxHalfTrack = 68
diskBitCycle = 4 // There is a dataLatch bit transferred every 4 cycles
diskLatchReadCycles = 7 // Loaded data is available for a little more than 7ns
diskWriteByteCycle = 32 // Load data to write every 32 cycles
diskWriteSelfSyncCycle = 40 // Save $FF every 40 cycles. Self sync is 10 bits: 1111 1111 00
diskMotorStartMs = 150 // Time with the disk spinning to get full speed
)
2020-10-14 21:54:51 +02:00
// NewCardDisk2 creates a new CardDisk2
func NewCardDisk2() *CardDisk2 {
var c CardDisk2
c.name = "Disk II"
c.loadRomFromResource("<internal>/DISK2.rom")
return &c
}
// GetInfo returns smartport info
func (c *CardDisk2) GetInfo() map[string]string {
info := make(map[string]string)
info["rom"] = "16 sector"
info["D1 name"] = c.drive[0].name
info["D1 track"] = strconv.FormatFloat(float64(c.drive[0].tracksStep)/4, 'f', 2, 64)
info["D1 power"] = strconv.FormatBool(c.drive[0].power)
info["D2 name"] = c.drive[1].name
info["D2 track"] = strconv.FormatFloat(float64(c.drive[1].tracksStep)/4, 'f', 2, 64)
info["D1 power"] = strconv.FormatBool(c.drive[1].power)
return info
}
func (c *CardDisk2) assign(a *Apple2, slot int) {
// Q1, Q2, Q3 and Q4 phase control soft switches,
2019-10-21 00:00:42 +02:00
for i := uint8(0); i < 4; i++ {
phase := i
2019-10-21 00:00:42 +02:00
c.addCardSoftSwitchR(phase<<1, func(_ *ioC0Page) uint8 {
// Update magnets and position
drive := &c.drive[c.selected]
drive.phases &^= (1 << phase)
drive.tracksStep = moveStep(drive.phases, drive.tracksStep)
return c.dataLatch // All even addresses return the last dataLatch
2019-10-21 00:00:42 +02:00
}, fmt.Sprintf("PHASE%vOFF", phase))
c.addCardSoftSwitchR((phase<<1)+1, func(_ *ioC0Page) uint8 {
// Update magnets and position
drive := &c.drive[c.selected]
drive.phases |= (1 << phase)
drive.tracksStep = moveStep(drive.phases, drive.tracksStep)
return 0
2020-09-23 18:14:45 +02:00
}, fmt.Sprintf("PHASE%vON", phase))
2019-03-02 20:41:25 +01:00
}
// Q4, power switch
2019-10-21 00:00:42 +02:00
c.addCardSoftSwitchR(0x8, func(_ *ioC0Page) uint8 {
2019-12-21 12:50:40 +01:00
drive := &c.drive[c.selected]
if drive.power {
drive.power = false
c.a.releaseFastMode()
if drive.diskette != nil {
drive.diskette.powerOff(c.a.cpu.GetCycles())
}
}
return c.dataLatch
2019-10-21 00:00:42 +02:00
}, "Q4DRIVEOFF")
c.addCardSoftSwitchR(0x9, func(_ *ioC0Page) uint8 {
2019-12-21 12:50:40 +01:00
drive := &c.drive[c.selected]
if !drive.power {
drive.power = true
c.a.requestFastMode()
if drive.diskette != nil {
drive.diskette.powerOn(c.a.cpu.GetCycles())
}
}
2019-03-02 20:41:25 +01:00
return 0
2019-12-21 12:50:40 +01:00
}, "Q4DRIVEON")
// Q5, drive selecion
2019-10-21 00:00:42 +02:00
c.addCardSoftSwitchR(0xA, func(_ *ioC0Page) uint8 {
2019-03-02 20:41:25 +01:00
c.selected = 0
return c.dataLatch
2019-10-21 00:00:42 +02:00
}, "Q5SELECT1")
c.addCardSoftSwitchR(0xB, func(_ *ioC0Page) uint8 {
2019-03-02 20:41:25 +01:00
c.selected = 1
return 0
2019-10-21 00:00:42 +02:00
}, "Q5SELECT2")
2019-03-02 20:41:25 +01:00
// Q6, Q7
2019-10-21 00:00:42 +02:00
for i := uint8(0xC); i <= 0xF; i++ {
iCopy := i
2019-10-21 00:00:42 +02:00
c.addCardSoftSwitchR(iCopy, func(_ *ioC0Page) uint8 {
return c.softSwitchQ6Q7(iCopy, 0)
2019-10-21 00:00:42 +02:00
}, "Q6Q7")
c.addCardSoftSwitchW(iCopy, func(_ *ioC0Page, value uint8) {
c.softSwitchQ6Q7(iCopy, value)
2019-10-21 00:00:42 +02:00
}, "Q6Q7")
2019-03-03 23:54:43 +01:00
}
c.cardBase.assign(a, slot)
}
2019-03-03 23:54:43 +01:00
2019-12-09 20:40:00 +01:00
// Q6: shift/load
// Q7: read/write
2020-10-14 21:54:51 +02:00
func (c *CardDisk2) softSwitchQ6Q7(index uint8, in uint8) uint8 {
switch index {
case 0xC: // Q6L
c.q6 = false
case 0xD: // Q6H
c.q6 = true
2019-12-09 20:40:00 +01:00
case 0xE: // Q7L
c.q7 = false
case 0xF: // Q7H
c.q7 = true
}
c.processQ6Q7(in)
if index&1 == 0 {
// All even addresses return the last dataLatch
return c.dataLatch
}
return 0
}
2020-10-14 21:54:51 +02:00
func (c *CardDisk2) processQ6Q7(in uint8) {
d := &c.drive[c.selected]
if d.diskette == nil {
return
}
2019-12-09 20:40:00 +01:00
if !c.q6 { // shift
if !c.q7 { // Q6L-Q7L: Read
c.dataLatch = d.diskette.read(d.tracksStep, c.a.cpu.GetCycles())
} else { // Q6L-Q7H: Write the dataLatch value to disk. Shift data out
d.diskette.write(d.tracksStep, c.dataLatch, c.a.cpu.GetCycles())
}
2019-12-09 20:40:00 +01:00
} else { // load
if !c.q7 { // Q6H-Q7L: Sense write protect / prewrite state
// Bit 7 of the control status register means write protected
c.dataLatch = 0 // Never write protected
} else { // Q6H-Q7H: Load data into the controller
c.dataLatch = in
}
}
2019-12-21 11:16:07 +01:00
if c.dataLatch >= 0x80 {
//fmt.Printf("Datalacth: 0x%.2x in cycle %v\n", c.dataLatch, c.a.cpu.GetCycles())
}
}
/*
Stepper motor to position the track.
There are a number of group of four magnets. The stepper motor can be thought as a long
line of groups of magnets, each group on the same configuration. We call phase each of those
magnets. The cog is attracted to the enabled magnets, and can stay aligned to a magnet or
between two.
2020-10-07 09:37:13 +02:00
Phases (magnets): 3 2 1 0 3 2 1 0 3 2 1 0
Cog direction (step within a group): 7 6 5 4 3 2 1 0 7 6 5 4 3 2 1 0 7 6 5 4 3 2 1 0
2020-10-07 09:37:13 +02:00
We will consider that the cog would go to the prefferred position if there is one. Independently
of the previous position. The previous position is only used to know if it goes up or down
a full group.
*/
const (
undefinedPosition = -1
maxStep = 68 * 2 // What is the maximum quarter tracks a DiskII can go?
stepsPerGroup = 8
stepsPerTrack = 4
)
var cogPositions = []int{
undefinedPosition, // 0000, phases active
0, // 0001
2, // 0010
1, // 0011
4, // 0100
undefinedPosition, // 0101
3, // 0110
2, // 0111
6, // 1000
7, // 1001
undefinedPosition, // 1010
0, // 1011
5, // 1100
6, // 1101
4, // 1110
undefinedPosition, // 1111
}
func moveStep(phases uint8, prevStep int) int {
//fmt.Printf("magnets: 0x%x\n", phases)
cogPosition := cogPositions[phases]
if cogPosition == undefinedPosition {
// Don't move if magnets don't push on a defined direction.
return prevStep
}
prevPosition := prevStep % stepsPerGroup // Direction, step in the current group of magnets.
delta := cogPosition - prevPosition
if delta < 0 {
delta = delta + stepsPerGroup
}
var nextStep int
if delta < 4 {
// Steps up
nextStep = prevStep + delta
if nextStep > maxStep {
nextStep = maxStep
}
} else if delta == 4 {
2020-10-07 09:37:13 +02:00
// Don't move if magnets push on the opposite direction
nextStep = prevStep
} else { // delta > 4
// Steps down
nextStep = prevStep + delta - stepsPerGroup
if nextStep < 0 {
nextStep = 0
}
}
return nextStep
}
2020-10-14 21:54:51 +02:00
func (d *cardDisk2Drive) insertDiskette(name string, dt diskette) {
d.name = name
d.diskette = dt
2019-03-05 00:00:12 +01:00
}