more patchers

This commit is contained in:
4am 2018-05-27 10:24:01 -04:00
parent e924950634
commit d8f7960335
6 changed files with 203 additions and 50 deletions

View File

@ -139,7 +139,11 @@ class RWTS:
self.sector_order = sector_order self.sector_order = sector_order
self.nibble_translate_table = nibble_translate_table self.nibble_translate_table = nibble_translate_table
self.logger = logger or SilentLogger self.logger = logger or SilentLogger
self.track_num = 0
def seek(self, track_num):
self.track_num = track_num
def reorder_to_logical_sectors(self, sectors): def reorder_to_logical_sectors(self, sectors):
logical = {} logical = {}
for k, v in sectors.items(): for k, v in sectors.items():
@ -218,14 +222,17 @@ class RWTS:
# if we can't even find a single address prologue, just give up # if we can't even find a single address prologue, just give up
self.logger.debug("can't find a single address prologue so LGTM or whatever") self.logger.debug("can't find a single address prologue so LGTM or whatever")
break break
# for edd->woz conversion, only save some of the bits preceding
# the address prologue
if track.bit_index - start_bit_index > 256: if track.bit_index - start_bit_index > 256:
start_bit_index = track.bit_index - 256 start_bit_index = track.bit_index - 256
# decode address field # decode address field
address_field = self.address_field_at_point(track) address_field = self.address_field_at_point(track)
self.logger.debug("found sector %s" % hex(address_field.sector_num)[2:].upper())
if address_field.sector_num in verified_sectors: if address_field.sector_num in verified_sectors:
# the sector we just found is a sector we've already decoded # the sector we just found is a sector we've already decoded
# properly, so skip it # properly, so skip it
self.logger.debug("duplicate sector %d" % address_field.sector_num) self.logger.debug("duplicate sector %d, continuing" % address_field.sector_num)
continue continue
if address_field.sector_num > self.sectors_per_track: if address_field.sector_num > self.sectors_per_track:
# found a weird sector whose ID is out of range # found a weird sector whose ID is out of range
@ -240,21 +247,16 @@ class RWTS:
# verifying the address field epilogue failed, but this is # verifying the address field epilogue failed, but this is
# not necessarily fatal because there might be another copy # not necessarily fatal because there might be another copy
# of this sector later # of this sector later
self.logger.debug("verify_address_epilogue_at_point failed, continuing")
continue continue
if not self.find_data_prologue(track): if not self.find_data_prologue(track):
# if we can't find a data field prologue, just give up # if we can't find a data field prologue, just give up
self.logger.debug(repr(self.data_prologue)) self.logger.debug("find_data_prologue failed, giving up")
break break
# read and decode the data field, and verify the data checksum # read and decode the data field, and verify the data checksum
decoded = self.data_field_at_point(track) decoded = self.data_field_at_point(track)
if not decoded: if not decoded:
self.logger.debug("data_field_at_point failed") self.logger.debug("data_field_at_point failed, continuing")
# if DEBUG and address_field.sector_num == 0x0A:
# DEBUG_CACHE.append(track.bits[start_bit_index:track.bit_index])
# if len(DEBUG_CACHE) == 2:
# import code
# cache = DEBUG_CACHE
# code.interact(local=locals())
# decoding data field failed, but this is not necessarily fatal # decoding data field failed, but this is not necessarily fatal
# because there might be another copy of this sector later # because there might be another copy of this sector later
continue continue
@ -331,44 +333,54 @@ class UniversalRWTSIgnoreEpilogues(UniversalRWTS):
class DOS33RWTS(RWTS): class DOS33RWTS(RWTS):
def __init__(self, logical_sectors, logger): def __init__(self, logical_sectors, logger):
address_prologue = (logical_sectors[3][0x55], self.reset(logical_sectors)
logical_sectors[3][0x5F],
logical_sectors[3][0x6A])
address_epilogue = (logical_sectors[3][0x91],
logical_sectors[3][0x9B])
data_prologue = (logical_sectors[2][0xE7],
logical_sectors[2][0xF1],
logical_sectors[2][0xFC])
data_epilogue = (logical_sectors[3][0x35],
logical_sectors[3][0x3F])
nibble_translate_table = {}
for nibble in range(0x96, 0x100):
nibble_translate_table[nibble] = logical_sectors[4][nibble]
RWTS.__init__(self, RWTS.__init__(self,
sectors_per_track=16, sectors_per_track=16,
address_prologue=address_prologue, address_prologue=self.address_prologue,
address_epilogue=address_epilogue, address_epilogue=self.address_epilogue,
data_prologue=data_prologue, data_prologue=self.data_prologue,
data_epilogue=data_epilogue, data_epilogue=self.data_epilogue,
nibble_translate_table=nibble_translate_table, nibble_translate_table=self.nibble_translate_table,
logger=logger) logger=logger)
class D5TimingBitRWTS(RWTS): def reset(self, logical_sectors):
def __init__(self, logical_sectors, logger): self.address_prologue = (logical_sectors[3][0x55],
data_prologue = (logical_sectors[2][0xE7], logical_sectors[3][0x5F],
0xAA, logical_sectors[3][0x6A])
logical_sectors[2][0xFC]) self.address_epilogue = (logical_sectors[3][0x91],
data_epilogue = (logical_sectors[3][0x35], logical_sectors[3][0x9B])
0xAA) self.data_prologue = (logical_sectors[2][0xE7],
nibble_translate_table = {} logical_sectors[2][0xF1],
logical_sectors[2][0xFC])
self.data_epilogue = (logical_sectors[3][0x35],
logical_sectors[3][0x3F])
self.nibble_translate_table = {}
for nibble in range(0x96, 0x100): for nibble in range(0x96, 0x100):
nibble_translate_table[nibble] = logical_sectors[4][nibble] self.nibble_translate_table[nibble] = logical_sectors[4][nibble]
RWTS.__init__(self,
sectors_per_track=16, class BorderRWTS(DOS33RWTS):
data_prologue=data_prologue, # TODO doesn't work yet, not sure why
data_epilogue=data_epilogue, def reset(self, logical_sectors):
nibble_translate_table=nibble_translate_table, DOS33RWTS.reset(self, logical_sectors)
logger=logger) self.address_prologue = (logical_sectors[9][0x16],
logical_sectors[9][0x1B],
logical_sectors[9][0x20])
self.address_epilogue = (logical_sectors[9][0x25],
logical_sectors[9][0x2A])
self.data_prologue = (logical_sectors[8][0xFD],
logical_sectors[9][0x02],
logical_sectors[9][0x02])
self.data_epilogue = (logical_sectors[9][0x0C],
logical_sectors[9][0x11])
class D5TimingBitRWTS(DOS33RWTS):
def reset(self, logical_sectors):
DOS33RWTS.reset(self, logical_sectors)
self.data_prologue = (logical_sectors[2][0xE7],
0xAA,
logical_sectors[2][0xFC])
self.data_epilogue = (logical_sectors[3][0x35],
0xAA)
def find_address_prologue(self, track): def find_address_prologue(self, track):
starting_revolutions = track.revolutions starting_revolutions = track.revolutions
@ -400,7 +412,7 @@ class BasePassportProcessor: # base class
#JMPB660Patcher, #JMPB660Patcher,
#JMPB720Patcher, #JMPB720Patcher,
bademu.BadEmuPatcher, bademu.BadEmuPatcher,
#BadEmu2Patcher, bademu2.BadEmu2Patcher,
rwts.RWTSPatcher, rwts.RWTSPatcher,
#RWTSLogPatcher, #RWTSLogPatcher,
#MECC1Patcher, #MECC1Patcher,
@ -412,7 +424,7 @@ class BasePassportProcessor: # base class
#DavidBB03Patcher, #DavidBB03Patcher,
#RWTSSwapPatcher, #RWTSSwapPatcher,
#RWTSSwap2Patcher, #RWTSSwap2Patcher,
#BorderPatcher, border.BorderPatcher,
#JMPAE8EPatcher, #JMPAE8EPatcher,
#JMPBBFEPatcher, #JMPBBFEPatcher,
#DatasoftPatcher, #DatasoftPatcher,
@ -437,7 +449,7 @@ class BasePassportProcessor: # base class
#ProDOSRWTSPatcher, #ProDOSRWTSPatcher,
#ProDOS6APatcher, #ProDOS6APatcher,
#ProDOSMECCPatcher, #ProDOSMECCPatcher,
#BBF9Patcher, bbf9.BBF9Patcher,
#MemoryConfigPatcher, #MemoryConfigPatcher,
#OriginPatcher, #OriginPatcher,
#RWTSSwapMECCPatcher, #RWTSSwapMECCPatcher,
@ -450,7 +462,7 @@ class BasePassportProcessor: # base class
#EAPatcher, #EAPatcher,
#GamcoPatcher, #GamcoPatcher,
#OptimumPatcher, #OptimumPatcher,
#BootCounterPatcher, bootcounter.BootCounterPatcher,
#JMPB412Patcher, #JMPB412Patcher,
#JMPB400Patcher, #JMPB400Patcher,
advint.AdventureInternationalPatcher, advint.AdventureInternationalPatcher,
@ -680,6 +692,8 @@ class BasePassportProcessor: # base class
else: else:
self.logger.PrintByID("dos33boot0") self.logger.PrintByID("dos33boot0")
logical_sectors = temporary_rwts_for_t00.reorder_to_logical_sectors(physical_sectors) logical_sectors = temporary_rwts_for_t00.reorder_to_logical_sectors(physical_sectors)
if border.BorderPatcher(self.g).run(logical_sectors, 0):
return BorderRWTS(logical_sectors, self.logger)
return self.TraceDOS33(logical_sectors) return self.TraceDOS33(logical_sectors)
# TODO JSR08B3 # TODO JSR08B3
# TODO MECC fastloader # TODO MECC fastloader
@ -733,10 +747,9 @@ class BasePassportProcessor: # base class
# LDX $1FE8 e.g. Pinball Construction Set (1983) # LDX $1FE8 e.g. Pinball Construction Set (1983)
use_builtin = find.at(0x43, logical_sectors[8], b'\xAE\xE8\x1F') use_builtin = find.at(0x43, logical_sectors[8], b'\xAE\xE8\x1F')
if not use_builtin: if not use_builtin:
# check for D5+timing+bit RWTS # check for D5+timingbit RWTS
if find.at(0x58, logical_sectors[3], b'\xEA\xBD\x8C\xC0\xC9\xD5'): if find.at(0x59, logical_sectors[3], b'\xBD\x8C\xC0\xC9\xD5'):
self.logger.PrintByID("diskrwts") self.logger.PrintByID("diskrwts")
self.g.is_rwts = True
return D5TimingBitRWTS(logical_sectors, self.logger) return D5TimingBitRWTS(logical_sectors, self.logger)
# TODO handle Milliken here # TODO handle Milliken here
@ -747,7 +760,6 @@ class BasePassportProcessor: # base class
return self.StartWithUniv() return self.StartWithUniv()
self.logger.PrintByID("diskrwts") self.logger.PrintByID("diskrwts")
self.g.is_rwts = True
return DOS33RWTS(logical_sectors, self.logger) return DOS33RWTS(logical_sectors, self.logger)
def StartWithUniv(self): def StartWithUniv(self):
@ -780,6 +792,7 @@ class BasePassportProcessor: # base class
# main loop - loop through disk from track $22 down to track $00 # main loop - loop through disk from track $22 down to track $00
for track_num in range(0x22, -1, -1): for track_num in range(0x22, -1, -1):
self.g.track = track_num self.g.track = track_num
self.rwts.seek(track_num)
self.logger.debug("Seeking to track %s" % hex(self.g.track)) self.logger.debug("Seeking to track %s" % hex(self.g.track))
try_again = True try_again = True
while try_again: while try_again:
@ -787,6 +800,8 @@ class BasePassportProcessor: # base class
physical_sectors = self.rwts.decode_track(self.tracks[track_num], self.burn) physical_sectors = self.rwts.decode_track(self.tracks[track_num], self.burn)
if len(physical_sectors) == self.rwts.sectors_per_track: if len(physical_sectors) == self.rwts.sectors_per_track:
continue continue
else:
self.logger.debug("found %d sectors" % len(physical_sectors))
if (0x0F not in physical_sectors) and self.SkipTrack(track_num, self.tracks[track_num]): if (0x0F not in physical_sectors) and self.SkipTrack(track_num, self.tracks[track_num]):
physical_sectors = None physical_sectors = None
continue continue
@ -815,11 +830,41 @@ class BasePassportProcessor: # base class
pass pass
class Verify(BasePassportProcessor): class Verify(BasePassportProcessor):
def AnalyzeT00(self, logical_sectors):
self.g.is_boot1 = find.at(0x00, logical_sectors[1],
b'\x8E\xE9\xB7\x8E\xF7\xB7\xA9\x01'
b'\x8D\xF8\xB7\x8D\xEA\xB7\xAD\xE0'
b'\xB7\x8D\xE1\xB7\xA9\x02\x8D\xEC'
b'\xB7\xA9\x04\x8D\xED\xB7\xAC\xE7'
b'\xB7\x88\x8C\xF1\xB7\xA9\x01\x8D'
b'\xF4\xB7\x8A\x4A\x4A\x4A\x4A\xAA'
b'\xA9\x00\x9D\xF8\x04\x9D\x78\x04')
self.g.is_master = find.at(0x00, logical_sectors[1],
b'\x8E\xE9\x37\x8E\xF7\x37\xA9\x01'
b'\x8D\xF8\x37\x8D\xEA\x37\xAD\xE0'
b'\x37\x8D\xE1\x37\xA9\x02\x8D\xEC'
b'\x37\xA9\x04\x8D\xED\x37\xAC\xE7'
b'\x37\x88\x8C\xF1\x37\xA9\x01\x8D'
b'\xF4\x37\x8A\x4A\x4A\x4A\x4A\xAA'
b'\xA9\x00\x9D\xF8\x04\x9D\x78\x04')
self.g.is_rwts = find.wild_at(0x00, logical_sectors[7],
b'\x84\x48\x85\x49\xA0\x02\x8C' + find.WILDCARD + \
find.WILDCARD + b'\xA0\x04\x8C' + find.WILDCARD + find.WILDCARD + b'\xA0\x01' + \
b'\xB1\x48\xAA\xA0\x0F\xD1\x48\xF0'
b'\x1B\x8A\x48\xB1\x48\xAA\x68\x48'
b'\x91\x48\xBD\x8E\xC0\xA0\x08\xBD'
b'\x8C\xC0\xDD\x8C\xC0\xD0\xF6\x88'
b'\xD0\xF8\x68\xAA\xBD\x8E\xC0\xBD'
b'\x8C\xC0\xA0\x08\xBD\x8C\xC0\x48')
def save_track(self, track_num, physical_sectors): def save_track(self, track_num, physical_sectors):
if not physical_sectors: return {} if not physical_sectors: return {}
logical_sectors = self.rwts.reorder_to_logical_sectors(physical_sectors) logical_sectors = self.rwts.reorder_to_logical_sectors(physical_sectors)
should_run_patchers = (len(physical_sectors) == 16) # TODO should_run_patchers = (len(physical_sectors) == 16) # TODO
if should_run_patchers: if should_run_patchers:
if track_num == 0:
# set additional globals for patchers to use
self.AnalyzeT00(logical_sectors)
for patcher in self.patchers: for patcher in self.patchers:
if patcher.should_run(track_num): if patcher.should_run(track_num):
patches = patcher.run(logical_sectors, track_num) patches = patcher.run(logical_sectors, track_num)

View File

@ -3,6 +3,10 @@ __all__ = [
"a6bc95", "a6bc95",
"advint", "advint",
"bademu", "bademu",
"bademu2",
"bbf9",
"bootcounter",
"border",
"d5d5f7", "d5d5f7",
"microfun", "microfun",
"rwts", "rwts",

View File

@ -0,0 +1,25 @@
from passport.patchers import Patch, Patcher
from passport.util import *
class BadEmu2Patcher(Patcher):
"""RWTS checks for timing bit by checking if data latch is still $D5 after waiting "too long" but this confuses legacy emulators (AppleWin, older versions of MAME) so we patch it for compatibility
tested on
- Dino Dig
- Make A Face
"""
def should_run(self, track_num):
return self.g.is_rwts and (track_num == 0)
def run(self, logical_sectors, track_num):
if not find.at(0x4F, logical_sectors[3],
b'\xBD\x8C\xC0'
b'\x10\xFB'
b'\x4A'
b'\xC9\x6A'
b'\xD0\xEF'
b'\xBD\x8C\xC0'
b'\xC9\xD5'
b'\xF0\x12'): return []
return [Patch(0, 3, 0x59, b'\xF0\x05')]
return patches

32
passport/patchers/bbf9.py Normal file
View File

@ -0,0 +1,32 @@
from passport.patchers import Patch, Patcher
from passport.util import *
class BBF9Patcher(Patcher):
"""patch nibble check seen in Sunburst disks 1988 and later
see write-up of 4am crack no. 1165 Muppet Slate
tested on
- Muppet Slate (1988)
- Memory Building Blocks (1989)
- Odd One Out (1989)
- Regrouping (1989)
- Simon Says (1989)
- Teddy and Iggy (1990)
- 1-2-3 Sequence Me (1991)
"""
def should_run(self, track_num):
return self.g.is_prodos
def run(self, logical_sectors, track_num):
buffy = concat_track(logical_sectors)
if -1 == find.wild(buffy,
b'\x8E\xC0'
b'\x18'
b'\xA5' + find.WILDCARD + \
b'\x69\x8C'
b'\x8D'): return []
offset = find.wild(buffy,
b'\xBD\x89\xC0')
if offset == -1: return []
return [Patch(track_num, offset // 256, offset % 256, b'\x18\x60', "bbf9")]

View File

@ -0,0 +1,25 @@
from passport.patchers import Patch, Patcher
from passport.util import *
class BootCounterPatcher(Patcher):
"""MECC "limited backup" disks contain code to self-destruct after a certain number of boots"""
def should_run(self, track_num):
return track_num == 1
def run(self, logical_sectors, track_num):
if not find.wild_at(0x00, logical_sectors[0],
b'\xAD\xF3\x03'
b'\x8D\xF4\x03'
b'\x20\x2F\xFB'
b'\x20\x93\xFE'
b'\x20\x89\xFE'
b'\x20\x58\xFC'
b'\xA9\x0A'
b'\x85\x25'
b'\x2C' + find.WILDCARD + find.WILDCARD + \
b'\xCE\x17\x18'
b'\xD0\x05'
b'\xA9\x80'
b'\x8D\x18\x18'): return []
return [Patch(1, 0, 0x00, b'\x4C\x03\x1B', "bootcounter")]
return patches

View File

@ -0,0 +1,22 @@
from passport.patchers import Patch, Patcher
from passport.util import *
class BorderPatcher(Patcher):
"""RWTS changes prologue and epilogue sequences with an RWTS swapper at $BE5A
tested on
- Arena
- Early Bird
"""
def should_run(self, track_num):
return self.g.is_boot0 and self.g.is_boot1 and track_num == 0
def run(self, logical_sectors, track_num):
if not find.at(0x5A, logical_sectors[8],
b'\xC9\x23'
b'\xB0\xEB'
b'\x0A'
b'\x20\x6C\xBF'
b'\xEA'
b'\xEA'): return []
return [Patch(0, 8, 0x5A, b'\x48\xA0\x01\xB1\x3C\x6A\x68\x90\x08\x0A', "border")]