2019-01-29 19:31:10 -05:00
|
|
|
from passport.wozardry import Track, raise_if
|
2018-09-10 10:19:46 -04:00
|
|
|
from passport import a2rchery
|
|
|
|
import bitarray
|
|
|
|
import collections
|
|
|
|
|
2019-01-29 19:31:10 -05:00
|
|
|
class A2RSeekError(a2rchery.A2RError): pass
|
|
|
|
|
2018-09-10 10:19:46 -04:00
|
|
|
class A2RImage:
|
|
|
|
def __init__(self, filename=None, stream=None):
|
|
|
|
self.filename = filename
|
|
|
|
self.tracks = collections.OrderedDict()
|
|
|
|
self.a2r_image = a2rchery.A2RReader(filename, stream)
|
2019-02-17 18:55:01 -05:00
|
|
|
self.speed = 32
|
2018-09-10 10:19:46 -04:00
|
|
|
|
|
|
|
def to_bits(self, flux_record):
|
|
|
|
"""|flux_record| is a dictionary of 'capture_type', 'data_length', 'tick_count', and 'data'"""
|
|
|
|
bits = bitarray.bitarray()
|
2019-01-29 18:30:21 -05:00
|
|
|
if not flux_record or flux_record["capture_type"] != a2rchery.kCaptureTiming:
|
2019-02-02 08:33:39 -05:00
|
|
|
return bits
|
|
|
|
fluxxen = flux_record["data"][1:]
|
|
|
|
if not self.speed:
|
|
|
|
speeds = [(len([1 for i in fluxxen[:8192] if i%t==0]), t) for t in range(0x1e,0x23)]
|
|
|
|
speeds.sort()
|
|
|
|
self.speed = speeds[-1][1]
|
|
|
|
speed = self.speed
|
|
|
|
flux_total = flux_start = -speed//2
|
|
|
|
for flux_value in fluxxen:
|
2018-09-10 10:19:46 -04:00
|
|
|
flux_total += flux_value
|
|
|
|
if flux_value == 0xFF:
|
|
|
|
continue
|
2019-02-02 08:33:39 -05:00
|
|
|
if flux_total >= speed:
|
|
|
|
bits.extend("0" * (flux_total // speed))
|
|
|
|
bits.extend("1")
|
|
|
|
flux_total = flux_start
|
|
|
|
return bits
|
2019-01-29 18:30:21 -05:00
|
|
|
|
2018-09-10 10:19:46 -04:00
|
|
|
def seek(self, track_num):
|
|
|
|
if type(track_num) != float:
|
|
|
|
track_num = float(track_num)
|
|
|
|
if track_num < 0.0 or \
|
|
|
|
track_num > 35.0 or \
|
|
|
|
track_num.as_integer_ratio()[1] not in (1,2,4):
|
2019-01-29 19:31:10 -05:00
|
|
|
raise A2RSeekError("Invalid track %s" % track_num)
|
2018-09-10 10:19:46 -04:00
|
|
|
location = int(track_num * 4)
|
|
|
|
if not self.tracks.get(location):
|
2019-02-01 09:22:45 -05:00
|
|
|
# just return the bits from the first flux read
|
|
|
|
# (if the caller determines that they're not good, it will call reseek()
|
|
|
|
# which is smarter but takes longer)
|
|
|
|
bits = bitarray.bitarray()
|
2019-02-02 08:33:39 -05:00
|
|
|
if location in self.a2r_image.flux:
|
|
|
|
bits = self.to_bits(self.a2r_image.flux[location][0])
|
2019-02-01 09:22:45 -05:00
|
|
|
self.tracks[location] = Track(bits, len(bits))
|
|
|
|
return self.tracks[location]
|
|
|
|
|
|
|
|
def reseek(self, track_num):
|
|
|
|
location = int(track_num * 4)
|
2019-02-02 08:33:39 -05:00
|
|
|
# reset cached speed so we'll recalculate it
|
|
|
|
self.speed = 0
|
|
|
|
# read the rest of the flux records and concatenate them on the end
|
|
|
|
# of the existing bitstream (this assumes you've called seek() before
|
|
|
|
# on this track)
|
|
|
|
all_bits = self.tracks[location].bits
|
|
|
|
for flux_record in self.a2r_image.flux[location][1:]:
|
|
|
|
all_bits.extend(self.to_bits(flux_record))
|
2019-02-01 09:22:45 -05:00
|
|
|
self.tracks[location] = Track(all_bits, len(all_bits))
|
2018-09-10 10:19:46 -04:00
|
|
|
return self.tracks[location]
|