mirror of
https://github.com/robmcmullen/atrcopy.git
synced 2025-01-16 18:32:35 +00:00
Initial support for DOS 3.3 VTOC
This commit is contained in:
parent
ac8b750c44
commit
a4726f1c5a
224
atrcopy/dos33.py
224
atrcopy/dos33.py
@ -8,6 +8,129 @@ import logging
|
|||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class Dos33Dirent(object):
|
||||||
|
format = np.dtype([
|
||||||
|
('track', 'u1'),
|
||||||
|
('sector', 'u1'),
|
||||||
|
('flags', 'u1'),
|
||||||
|
('name','S30'),
|
||||||
|
('num_sectors','<u2'),
|
||||||
|
])
|
||||||
|
|
||||||
|
def __init__(self, image, file_num=0, bytes=None):
|
||||||
|
self.file_num = file_num
|
||||||
|
self.flags = 0
|
||||||
|
self.locked = False
|
||||||
|
self.deleted = False
|
||||||
|
self.track = 0
|
||||||
|
self.sector = 0
|
||||||
|
self.filename = ""
|
||||||
|
self.num_sectors = 0
|
||||||
|
self.is_sane = True
|
||||||
|
self.current_sector_index = 0
|
||||||
|
self.current_read = 0
|
||||||
|
self.sectors_seen = None
|
||||||
|
self.sector_map = None
|
||||||
|
self.parse_raw_dirent(image, bytes)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
flags = self.summary()
|
||||||
|
return "File #%-2d (%s) %03d %-30s %03d %03d" % (self.file_num, flags, self.num_sectors, self.filename, self.track, self.sector)
|
||||||
|
|
||||||
|
type_map = {
|
||||||
|
0x0: "T",
|
||||||
|
0x1: "I",
|
||||||
|
0x2: "A",
|
||||||
|
0x4: "B",
|
||||||
|
0x8: "S",
|
||||||
|
0x10: "R",
|
||||||
|
0x20: "A",
|
||||||
|
0x40: "B",
|
||||||
|
}
|
||||||
|
|
||||||
|
def summary(self):
|
||||||
|
locked = "*" if self.flags else " "
|
||||||
|
f = self.flags & 0x7f
|
||||||
|
try:
|
||||||
|
file_type = self.type_map[f]
|
||||||
|
except KeyError:
|
||||||
|
file_type = "?"
|
||||||
|
flags = "%s%s" % (locked, file_type)
|
||||||
|
return flags
|
||||||
|
|
||||||
|
@property
|
||||||
|
def verbose_info(self):
|
||||||
|
return self.summary
|
||||||
|
|
||||||
|
def parse_raw_dirent(self, image, bytes):
|
||||||
|
if bytes is None:
|
||||||
|
return
|
||||||
|
values = bytes.view(dtype=self.format)[0]
|
||||||
|
self.track = values[0]
|
||||||
|
if self.track == 0xff:
|
||||||
|
self.deleted = True
|
||||||
|
else:
|
||||||
|
self.deleted = False
|
||||||
|
self.sector = values[1]
|
||||||
|
self.flags = values[2]
|
||||||
|
self.filename = (bytes[3:0x20] - 0x80).tostring().rstrip()
|
||||||
|
self.num_sectors = int(values[4])
|
||||||
|
self.is_sane = self.sanity_check(image)
|
||||||
|
|
||||||
|
def sanity_check(self, image):
|
||||||
|
if self.deleted:
|
||||||
|
return True
|
||||||
|
if self.track == 0:
|
||||||
|
return False
|
||||||
|
s = image.header.sector_from_track(self.track, self.sector)
|
||||||
|
if not image.header.sector_is_valid(s):
|
||||||
|
return False
|
||||||
|
if self.num_sectors < 0 or self.num_sectors > image.header.max_sectors:
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
|
def get_track_sector_list(self, image):
|
||||||
|
sector = image.header.sector_from_track(self.track, self.sector)
|
||||||
|
sector_list = []
|
||||||
|
while sector > 0:
|
||||||
|
image.assert_valid_sector(sector)
|
||||||
|
print "reading track/sector list", sector
|
||||||
|
values, style = image.get_sectors(sector)
|
||||||
|
sector = image.header.sector_from_track(values[1], values[2])
|
||||||
|
i = 0x0c
|
||||||
|
while i < 256:
|
||||||
|
t = values[i]
|
||||||
|
s = values[i + 1]
|
||||||
|
i += 2
|
||||||
|
if t == 0:
|
||||||
|
sector = 0
|
||||||
|
break
|
||||||
|
sector_list.append(image.header.sector_from_track(t, s))
|
||||||
|
self.sector_map = sector_list
|
||||||
|
|
||||||
|
def start_read(self, image):
|
||||||
|
if not self.is_sane:
|
||||||
|
raise InvalidDirent("Invalid directory entry '%s'" % str(self))
|
||||||
|
self.get_track_sector_list(image)
|
||||||
|
self.current_sector_index = 0
|
||||||
|
self.current_read = self.num_sectors
|
||||||
|
|
||||||
|
def read_sector(self, image):
|
||||||
|
sector = self.sector_map[self.current_sector_index]
|
||||||
|
last = (self.current_sector_index == len(self.sector_map) - 1)
|
||||||
|
raw, pos, size = image.get_raw_bytes(sector)
|
||||||
|
bytes, num_data_bytes = self.process_raw_sector(image, raw)
|
||||||
|
return bytes, last, pos, num_data_bytes
|
||||||
|
|
||||||
|
def process_raw_sector(self, image, raw):
|
||||||
|
self.current_sector_index += 1
|
||||||
|
num_bytes = len(raw)
|
||||||
|
return raw[0:num_bytes], num_bytes
|
||||||
|
|
||||||
|
def get_filename(self):
|
||||||
|
return self.filename
|
||||||
|
|
||||||
|
|
||||||
class Dos33Header(AtrHeader):
|
class Dos33Header(AtrHeader):
|
||||||
file_format = "DOS 3.3"
|
file_format = "DOS 3.3"
|
||||||
|
|
||||||
@ -15,6 +138,7 @@ class Dos33Header(AtrHeader):
|
|||||||
AtrHeader.__init__(self, None, 256, 0)
|
AtrHeader.__init__(self, None, 256, 0)
|
||||||
self.header_offset = 0
|
self.header_offset = 0
|
||||||
self.sector_order = range(16)
|
self.sector_order = range(16)
|
||||||
|
self.vtoc_sector = 17 * 16
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return "%s Disk Image (size=%d (%dx%db)" % (self.file_format, self.image_size, self.max_sectors, self.sector_size)
|
return "%s Disk Image (size=%d (%dx%db)" % (self.file_format, self.image_size, self.max_sectors, self.sector_size)
|
||||||
@ -42,9 +166,13 @@ class Dos33Header(AtrHeader):
|
|||||||
size = self.sector_size
|
size = self.sector_size
|
||||||
return pos, size
|
return pos, size
|
||||||
|
|
||||||
|
def sector_from_track(self, track, sector):
|
||||||
|
return track * 16 + sector
|
||||||
|
|
||||||
|
|
||||||
class Dos33DiskImage(DiskImageBase):
|
class Dos33DiskImage(DiskImageBase):
|
||||||
def __init__(self, rawdata, filename=""):
|
def __init__(self, rawdata, filename=""):
|
||||||
|
self.first_catalog = 0
|
||||||
DiskImageBase.__init__(self, rawdata, filename)
|
DiskImageBase.__init__(self, rawdata, filename)
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
@ -60,19 +188,108 @@ class Dos33DiskImage(DiskImageBase):
|
|||||||
if (magic == [1, 56, 176, 3]).all():
|
if (magic == [1, 56, 176, 3]).all():
|
||||||
raise InvalidDiskImage("ProDOS format found; not DOS 3.3 image")
|
raise InvalidDiskImage("ProDOS format found; not DOS 3.3 image")
|
||||||
swap_order = False
|
swap_order = False
|
||||||
data, style = self.get_sectors(17 * 16)
|
data, style = self.get_sectors(self.header.vtoc_sector)
|
||||||
if data[3] == 3:
|
if data[3] == 3:
|
||||||
if data[1] < 35 and data[2] < 16:
|
if data[1] < 35 and data[2] < 16:
|
||||||
data, style = self.get_sectors(17 * 16 + 14)
|
data, style = self.get_sectors(self.header.vtoc_sector + 14)
|
||||||
if data[2] != 13:
|
if data[2] != 13:
|
||||||
swap_order = True
|
swap_order = True
|
||||||
else:
|
else:
|
||||||
raise InvalidDiskImage("Invalid VTOC location for DOS 3.3")
|
raise InvalidDiskImage("Invalid VTOC location for DOS 3.3")
|
||||||
|
|
||||||
print "swap", swap_order
|
print "swap", swap_order
|
||||||
|
|
||||||
|
vtoc_type = np.dtype([
|
||||||
|
('unused1', 'S1'),
|
||||||
|
('cat_track','u1'),
|
||||||
|
('cat_sector','u1'),
|
||||||
|
('dos_release', 'u1'),
|
||||||
|
('unused2', 'S2'),
|
||||||
|
('vol_num', 'u1'),
|
||||||
|
('unused3', 'S32'),
|
||||||
|
('max_pairs', 'u1'),
|
||||||
|
('unused4', 'S8'),
|
||||||
|
('last_track', 'u1'),
|
||||||
|
('track_dir', 'i1'),
|
||||||
|
('unused5', 'S2'),
|
||||||
|
('num_tracks', 'u1'),
|
||||||
|
('sectors_per_track', 'u1'),
|
||||||
|
('bytes_per_sector', 'u2'),
|
||||||
|
])
|
||||||
|
|
||||||
|
def get_vtoc(self):
|
||||||
|
data, style = self.get_sectors(self.header.vtoc_sector)
|
||||||
|
values = data[0:56].view(dtype=self.vtoc_type)[0]
|
||||||
|
self.first_catalog = self.header.sector_from_track(values[1], values[2])
|
||||||
|
self.assert_valid_sector(self.first_catalog)
|
||||||
|
self.total_sectors = int(values['num_tracks']) * int(values['sectors_per_track'])
|
||||||
|
self.dos_release = values['dos_release']
|
||||||
|
|
||||||
|
def get_directory(self):
|
||||||
|
sector = self.first_catalog
|
||||||
|
num = 0
|
||||||
|
files = []
|
||||||
|
while sector > 0:
|
||||||
|
self.assert_valid_sector(sector)
|
||||||
|
print "reading catalog sector", sector
|
||||||
|
values, style = self.get_sectors(sector)
|
||||||
|
sector = self.header.sector_from_track(values[1], values[2])
|
||||||
|
i = 0xb
|
||||||
|
while i < 256:
|
||||||
|
dirent = Dos33Dirent(self, num, values[i:i+0x23])
|
||||||
|
if not dirent.is_sane:
|
||||||
|
sector = 0
|
||||||
|
break
|
||||||
|
files.append(dirent)
|
||||||
|
print dirent
|
||||||
|
i += 0x23
|
||||||
|
num += 1
|
||||||
|
self.files = files
|
||||||
|
|
||||||
def get_boot_segments(self):
|
def get_boot_segments(self):
|
||||||
return []
|
return []
|
||||||
|
|
||||||
|
def get_vtoc_segments(self):
|
||||||
|
r = self.rawdata
|
||||||
|
segments = []
|
||||||
|
addr = 0
|
||||||
|
start, count = self.get_contiguous_sectors(self.header.vtoc_sector, 1)
|
||||||
|
segment = RawSectorsSegment(r[start:start+count], self.header.vtoc_sector, 1, count, 0, 0, self.header.sector_size, name="VTOC")
|
||||||
|
segments.append(segment)
|
||||||
|
return segments
|
||||||
|
|
||||||
|
def get_directory_segments(self):
|
||||||
|
byte_order = []
|
||||||
|
r = self.rawdata
|
||||||
|
segments = []
|
||||||
|
sector = self.first_catalog
|
||||||
|
while sector > 0:
|
||||||
|
self.assert_valid_sector(sector)
|
||||||
|
print "reading catalog sector", sector
|
||||||
|
raw, pos, size = self.get_raw_bytes(sector)
|
||||||
|
byte_order.extend(range(pos, pos + size))
|
||||||
|
sector = self.header.sector_from_track(raw[1], raw[2])
|
||||||
|
raw = self.rawdata.get_indexed(byte_order)
|
||||||
|
segment = DefaultSegment(raw, name="Catalog")
|
||||||
|
segments.append(segment)
|
||||||
|
return segments
|
||||||
|
|
||||||
|
def get_file_segment(self, dirent):
|
||||||
|
byte_order = []
|
||||||
|
dirent.start_read(self)
|
||||||
|
while True:
|
||||||
|
bytes, last, pos, size = dirent.read_sector(self)
|
||||||
|
byte_order.extend(range(pos, pos + size))
|
||||||
|
if last:
|
||||||
|
break
|
||||||
|
if len(byte_order) > 0:
|
||||||
|
name = "%s %ds@%d" % (dirent.get_filename(), dirent.num_sectors, dirent.sector_map[0])
|
||||||
|
verbose_name = "%s (%d sectors, first@%d) %s" % (dirent.get_filename(), dirent.num_sectors, dirent.sector_map[0], dirent.verbose_info)
|
||||||
|
raw = self.rawdata.get_indexed(byte_order)
|
||||||
|
segment = DefaultSegment(raw, name=name, verbose_name=verbose_name)
|
||||||
|
else:
|
||||||
|
segment = EmptySegment(self.rawdata, name=dirent.get_filename())
|
||||||
|
return segment
|
||||||
|
|
||||||
|
|
||||||
class ProdosHeader(Dos33Header):
|
class ProdosHeader(Dos33Header):
|
||||||
@ -82,7 +299,6 @@ class ProdosHeader(Dos33Header):
|
|||||||
class ProdosDiskImage(Dos33DiskImage):
|
class ProdosDiskImage(Dos33DiskImage):
|
||||||
def read_header(self):
|
def read_header(self):
|
||||||
self.header = ProdosHeader()
|
self.header = ProdosHeader()
|
||||||
print "HI"
|
|
||||||
|
|
||||||
def get_boot_sector_info(self):
|
def get_boot_sector_info(self):
|
||||||
# based on logic from a2server
|
# based on logic from a2server
|
||||||
@ -101,5 +317,3 @@ class ProdosDiskImage(Dos33DiskImage):
|
|||||||
swap_order = True
|
swap_order = True
|
||||||
else:
|
else:
|
||||||
raise InvalidDiskImage("No ProDOS header info found")
|
raise InvalidDiskImage("No ProDOS header info found")
|
||||||
|
|
||||||
print "swap", swap_order
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user