2018-01-20 00:05:04 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/sha256"
|
|
|
|
"encoding/hex"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/paleotronic/diskm8/disk"
|
|
|
|
"github.com/paleotronic/diskm8/loggy"
|
|
|
|
)
|
|
|
|
|
|
|
|
func analyzeDOS16(id int, dsk *disk.DSKWrapper, info *Disk) {
|
|
|
|
|
|
|
|
l := loggy.Get(id)
|
|
|
|
|
|
|
|
// Sector bitmap
|
|
|
|
l.Logf("Reading Disk VTOC...")
|
|
|
|
vtoc, err := dsk.AppleDOSGetVTOC()
|
|
|
|
if err != nil {
|
|
|
|
l.Errorf("Error reading VTOC: %s", err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
info.Tracks, info.Sectors = vtoc.GetTracks(), vtoc.GetSectors()
|
|
|
|
|
|
|
|
if vtoc.BytesPerSector() != 256 {
|
|
|
|
l.Errorf("Disk does not seem to be AppleDOS - treat as generic")
|
|
|
|
dsk.Format = disk.GetDiskFormat(disk.DF_NONE)
|
|
|
|
analyzeNONE(id, dsk, info)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
l.Logf("Tracks: %d, Sectors: %d", info.Tracks, info.Sectors)
|
|
|
|
l.Logf("Sector order: %d", vtoc.GetTrackOrder())
|
|
|
|
|
|
|
|
l.Logf("Reading sector bitmap and SHA256'ing sectors")
|
|
|
|
|
|
|
|
info.Bitmap = make([]bool, info.Tracks*info.Sectors)
|
|
|
|
|
|
|
|
var useAlt bool
|
|
|
|
|
|
|
|
if vtoc.IsTSFree(17, 0) {
|
|
|
|
info.Bitmap, _ = dsk.AppleDOSUsedBitmap()
|
|
|
|
useAlt = true
|
|
|
|
}
|
|
|
|
|
|
|
|
info.ActiveSectors = make(DiskSectors, 0)
|
|
|
|
info.InactiveSectors = make(DiskSectors, 0)
|
|
|
|
|
|
|
|
activeData := make([]byte, 0)
|
|
|
|
|
|
|
|
for t := 0; t < info.Tracks; t++ {
|
|
|
|
|
|
|
|
for s := 0; s < info.Sectors; s++ {
|
|
|
|
|
|
|
|
if !useAlt {
|
|
|
|
info.Bitmap[t*info.Sectors+s] = !vtoc.IsTSFree(t, s)
|
|
|
|
}
|
|
|
|
|
|
|
|
// checksum sector
|
|
|
|
//info.SectorFingerprints[dsk.ChecksumSector(t, s)] = &DiskBlock{Track: t, Sector: s}
|
|
|
|
|
|
|
|
if info.Bitmap[t*info.Sectors+s] {
|
|
|
|
sector := &DiskSector{
|
|
|
|
Track: t,
|
|
|
|
Sector: s,
|
|
|
|
SHA256: dsk.ChecksumSector(t, s),
|
|
|
|
}
|
|
|
|
|
|
|
|
data := dsk.Read()
|
|
|
|
activeData = append(activeData, data...)
|
|
|
|
|
|
|
|
if *ingestMode&2 == 2 {
|
|
|
|
sector.Data = data
|
|
|
|
}
|
|
|
|
|
|
|
|
info.ActiveSectors = append(info.ActiveSectors, sector)
|
|
|
|
} else {
|
|
|
|
sector := &DiskSector{
|
|
|
|
Track: t,
|
|
|
|
Sector: s,
|
|
|
|
SHA256: dsk.ChecksumSector(t, s),
|
|
|
|
}
|
|
|
|
|
|
|
|
data := dsk.Read()
|
|
|
|
if *ingestMode&2 == 2 {
|
|
|
|
sector.Data = data
|
|
|
|
}
|
|
|
|
//activeData = append(activeData, data...)
|
|
|
|
|
|
|
|
info.InactiveSectors = append(info.InactiveSectors, sector)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
sum := sha256.Sum256(activeData)
|
|
|
|
info.SHA256Active = hex.EncodeToString(sum[:])
|
|
|
|
|
|
|
|
info.LogBitmap(id)
|
|
|
|
|
|
|
|
// Analyzing files
|
|
|
|
l.Log("Starting Analysis of files")
|
|
|
|
|
|
|
|
// lines := []string{
|
|
|
|
// "10 PRINT \"HELLO WORLDS\"",
|
|
|
|
// "20 GOTO 10",
|
|
|
|
// }
|
|
|
|
|
|
|
|
// e := dsk.AppleDOSWriteFile("CHEESE", disk.FileTypeAPP, disk.ApplesoftTokenize(lines), 0x801)
|
|
|
|
// if e != nil {
|
|
|
|
// l.Errorf("Error writing file: %s", e.Error())
|
|
|
|
// panic(e)
|
|
|
|
// }
|
|
|
|
// f, _ := os.Create("out.dsk")
|
|
|
|
// f.Write(dsk.Data)
|
|
|
|
// f.Close()
|
|
|
|
|
|
|
|
vtoc, files, err := dsk.AppleDOSGetCatalog("*")
|
|
|
|
if err != nil {
|
|
|
|
l.Errorf("Problem reading directory: %s", err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
info.Files = make([]*DiskFile, 0)
|
|
|
|
for _, fd := range files {
|
|
|
|
l.Logf("- Name=%s, Type=%s", fd.NameUnadorned(), fd.Type())
|
|
|
|
|
|
|
|
file := DiskFile{
|
|
|
|
Filename: fd.NameUnadorned(),
|
|
|
|
Type: fd.Type().String(),
|
|
|
|
Locked: fd.IsLocked(),
|
|
|
|
Ext: fd.Type().Ext(),
|
|
|
|
Created: time.Now(),
|
|
|
|
Modified: time.Now(),
|
|
|
|
}
|
|
|
|
|
|
|
|
//l.Log("start read")
|
2018-05-12 01:00:57 +00:00
|
|
|
size, addr, data, err := dsk.AppleDOSReadFileRaw(fd)
|
2018-01-20 00:05:04 +00:00
|
|
|
if err == nil {
|
|
|
|
sum := sha256.Sum256(data)
|
|
|
|
file.SHA256 = hex.EncodeToString(sum[:])
|
2018-05-12 01:00:57 +00:00
|
|
|
file.Size = size
|
2018-01-20 00:05:04 +00:00
|
|
|
if *ingestMode&1 == 1 {
|
|
|
|
if fd.Type() == disk.FileTypeAPP {
|
|
|
|
file.Text = disk.ApplesoftDetoks(data)
|
|
|
|
file.TypeCode = TypeMask_AppleDOS | TypeCode(fd.Type())
|
|
|
|
file.Data = data
|
|
|
|
file.LoadAddress = 0x801
|
|
|
|
} else if fd.Type() == disk.FileTypeINT {
|
|
|
|
file.Text = disk.IntegerDetoks(data)
|
|
|
|
file.TypeCode = TypeMask_AppleDOS | TypeCode(fd.Type())
|
|
|
|
file.LoadAddress = 0x1000
|
|
|
|
file.Data = data
|
|
|
|
} else if fd.Type() == disk.FileTypeTXT {
|
|
|
|
file.Text = disk.StripText(data)
|
|
|
|
file.Data = data
|
|
|
|
file.TypeCode = TypeMask_AppleDOS | TypeCode(fd.Type())
|
|
|
|
file.LoadAddress = 0x0000
|
|
|
|
} else if fd.Type() == disk.FileTypeBIN && len(data) >= 2 {
|
2018-05-12 01:00:57 +00:00
|
|
|
file.LoadAddress = addr
|
|
|
|
file.Data = data
|
2018-01-20 00:05:04 +00:00
|
|
|
file.TypeCode = TypeMask_AppleDOS | TypeCode(fd.Type())
|
|
|
|
} else {
|
|
|
|
file.LoadAddress = 0x0000
|
|
|
|
file.Data = data
|
|
|
|
file.TypeCode = TypeMask_AppleDOS | TypeCode(fd.Type())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//l.Log("end read")
|
|
|
|
|
|
|
|
info.Files = append(info.Files, &file)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
exists := exists(*baseName + "/" + info.GetFilename())
|
|
|
|
|
|
|
|
if !exists || *forceIngest {
|
|
|
|
info.WriteToFile(*baseName + "/" + info.GetFilename())
|
|
|
|
} else {
|
|
|
|
l.Log("Not writing as it already exists")
|
|
|
|
}
|
|
|
|
|
|
|
|
out(dsk.Format)
|
|
|
|
}
|