2021-05-09 19:48:54 +02:00
|
|
|
package izapple2
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"math/rand"
|
|
|
|
|
|
|
|
"github.com/ivanizag/izapple2/component"
|
|
|
|
"github.com/ivanizag/izapple2/storage"
|
|
|
|
)
|
|
|
|
|
|
|
|
type cardDisk2SequencerDrive struct {
|
|
|
|
data *storage.FileWoz
|
|
|
|
enabled bool
|
|
|
|
writeProtected bool
|
|
|
|
currentQuarterTrack int
|
|
|
|
|
|
|
|
position uint32 // Current position on the track
|
|
|
|
positionMax uint32 // As tracks may have different lengths position is related of positionMax of the las track
|
|
|
|
|
|
|
|
mc3470Buffer uint8 // Four bit buffer to detect weak bits and to add latency
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *cardDisk2SequencerDrive) insertDiskette(filename string) error {
|
2022-01-28 19:25:52 +01:00
|
|
|
data, writeable, err := LoadResource(filename)
|
2021-05-09 19:48:54 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
f, err := storage.NewFileWoz(data)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Discard not supported features
|
|
|
|
if f.Info.DiskType != 1 {
|
2021-08-05 21:12:52 +02:00
|
|
|
return errors.New("only 5.25 disks are supported")
|
2021-05-09 19:48:54 +02:00
|
|
|
}
|
|
|
|
if f.Info.BootSectorFormat == 2 { // Info not available in WOZ 1.0
|
2021-08-05 21:12:52 +02:00
|
|
|
return errors.New("woz 13 sector disks are not supported")
|
2021-05-09 19:48:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
d.data = f
|
|
|
|
d.writeProtected = !writeable
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *cardDisk2SequencerDrive) enable(enabled bool) {
|
|
|
|
d.enabled = enabled
|
|
|
|
}
|
|
|
|
|
2022-02-26 00:05:09 +01:00
|
|
|
func (d *cardDisk2SequencerDrive) moveHead(q0, q1, q2, q3 bool, trackTracer trackTracer) {
|
2021-05-09 19:48:54 +02:00
|
|
|
if !d.enabled {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
phases := component.PinsToByte([8]bool{
|
|
|
|
q0, q1, q2, q3,
|
|
|
|
false, false, false, false,
|
|
|
|
})
|
|
|
|
d.currentQuarterTrack = moveDriveStepper(phases, d.currentQuarterTrack)
|
2022-02-26 00:05:09 +01:00
|
|
|
|
|
|
|
if trackTracer != nil {
|
|
|
|
trackTracer.traceTrack(d.currentQuarterTrack)
|
|
|
|
}
|
2021-05-09 19:48:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (d *cardDisk2SequencerDrive) readPulse() bool {
|
|
|
|
if !d.enabled || d.data == nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get next bit taking into account the MC3470 latency and weak bits
|
2021-07-01 18:15:49 +02:00
|
|
|
var fluxBit bool
|
2021-05-09 19:48:54 +02:00
|
|
|
fluxBit, d.position, d.positionMax = d.data.GetNextBitAndPosition(
|
|
|
|
d.position,
|
|
|
|
d.positionMax,
|
|
|
|
d.currentQuarterTrack)
|
2021-07-01 18:15:49 +02:00
|
|
|
d.mc3470Buffer = (d.mc3470Buffer << 1) & 0x0f
|
|
|
|
if fluxBit {
|
|
|
|
d.mc3470Buffer++
|
|
|
|
}
|
2021-05-09 19:48:54 +02:00
|
|
|
bit := ((d.mc3470Buffer >> 1) & 0x1) != 0 // Use the previous to last bit to add latency
|
2021-05-10 22:13:55 +02:00
|
|
|
if d.mc3470Buffer == 0 && rand.Intn(100) < 30 {
|
2021-05-09 19:48:54 +02:00
|
|
|
// Four consecutive zeros. It'a a fake bit.
|
|
|
|
// Output a random value. 70% zero, 30% one
|
|
|
|
bit = true
|
|
|
|
}
|
|
|
|
|
|
|
|
return bit
|
|
|
|
}
|
|
|
|
|
2021-07-01 18:15:49 +02:00
|
|
|
func (d *cardDisk2SequencerDrive) writePulse(value bool) {
|
|
|
|
if d.writeProtected || !d.enabled || d.data == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
d.data.SetBit(
|
|
|
|
value,
|
|
|
|
d.position,
|
|
|
|
d.positionMax,
|
|
|
|
d.currentQuarterTrack)
|
2021-05-09 19:48:54 +02:00
|
|
|
}
|