Fixed VTOC bit packing and unpacking

This commit is contained in:
Rob McMullen 2017-02-23 12:04:14 -08:00
parent fe01a97c1f
commit 04edbed853

View File

@ -11,7 +11,7 @@ log = logging.getLogger(__name__)
class Dos33VTOC(VTOC): class Dos33VTOC(VTOC):
max_tracks = (256 - 0x38) / 4 # 50, but kept here in case sector size changed max_tracks = (256 - 0x38) / 4 # 50, but kept here in case sector size changed
max_sectors = max_tracks * 16 max_sectors = max_tracks * 16
reorder_index = np.tile(np.arange(7, -1, -1), max_tracks * 2) + (np.repeat(np.arange(max_tracks * 2), 8) * 8) vtoc_bit_reorder_index = np.tile(np.arange(15, -1, -1), max_tracks) + (np.repeat(np.arange(max_tracks), 16) * 16)
def parse_segments(self, segments): def parse_segments(self, segments):
# VTOC stored in groups of 4 bytes starting at 0x38 # VTOC stored in groups of 4 bytes starting at 0x38
@ -35,40 +35,41 @@ class Dos33VTOC(VTOC):
# i.e. each group of 16 bits needs to be reversed. # i.e. each group of 16 bits needs to be reversed.
self.vtoc = segments[0].data self.vtoc = segments[0].data
# create a view using little-endian 16 bit values, skipping every other # create a view starting at 0x38 where out of every 4 bytes, the first
# 16 bit value. This gets us 76543210FEDCBA98 76543210FEDCBA98 ... etc # two are used and the second 2 are skipped. Regular slicing doesn't
expanded = self.vtoc[0x38:].view(dtype="<u2") # work like this, so thanks to stackoverflow.com/questions/33801170,
used = expanded[0::2] # reshaping it to a 2d array with 4 elements in each row, doing a slice
# *there* to skip the last 2 entries in each row, then flattening it
# gives us what we need.
usedbytes = self.vtoc[0x38:].reshape((-1, 4))[:,:2].flatten()
# need to operate on a copy of the 16 bit values, because skipping # The bits here are still ordered backwards for each track, e.g. F E D
# every 2 and attempting to view as bytes results in a ValueError: "new # C B A 9 8 7 6 5 4 3 2 1 0
# type not compatible with array"
usedcopy = np.zeros([used.shape[0]], dtype="<u2")
usedcopy[:] = used
print repr(usedcopy)
# now convert to bytes, giving us 76543210 FEDCBA98 76543210 FEDCBA98
# ... etc, and unpack that to bits giving us a zero or one representing
# sectors 7 6 5 4 3 2 1 0 F E D ...
usedbytes = usedcopy.view(dtype=np.uint8)
bits = np.unpackbits(usedbytes) bits = np.unpackbits(usedbytes)
# and now we need to reverse the order of each group of 8 numbers # so we need to reorder them using numpy's indexing before stuffing
self.sector_map[0:self.max_sectors] = bits[reorder_index] # them into the sector map
self.sector_map[0:self.max_sectors] = bits[self.vtoc_bit_reorder_index]
log.debug("vtoc before: %s" % self.sector_map[0:35*16]) log.debug("vtoc before: %s" % self.sector_map[0:35*16])
def calc_bitmap(self): def calc_bitmap(self):
log.debug("vtoc after: %s" % self.sector_map[0:35*16]) log.debug("vtoc after: %s" % self.sector_map[0:35*16])
# bits inside bytes will be in the right order after this, but the # reverse the process from above, so swap the order of every 16 bits,
# bytes are still swapped and there are no gaps # turn them into bytes, then stuff them back into the vtoc. The bit
packed = np.packbits(self.sector_map[self.reorder_index]) # reorder list is commutative, so we don't need another order here.
print packed packed = np.packbits(self.sector_map[self.vtoc_bit_reorder_index])
vtoc = self.vtoc[0x38:].reshape((-1, 4))
print vtoc
packed = packed.reshape((-1, 2))
vtoc[:,:2] = packed[:,:]
print vtoc
# FIXME # FIXME
self.vtoc[0x38:] = packed self.vtoc[0x38:] = vtoc.flatten()
s = WriteableSector(self.bytes_per_sector, self.vtoc1) s = WriteableSector(self.bytes_per_sector, self.vtoc)
s.sector_num = 360 s.sector_num = 17 * 16
self.sectors.append(s) self.sectors.append(s)