nvidia-patch/win/tools/1337-diff/1337-diff.py

138 lines
4.3 KiB
Python
Raw Normal View History

2019-02-05 19:05:17 +00:00
#!/usr/bin/env python3
import sys
import argparse
import os.path
import itertools
PATCH_EXT = ".1337"
CRLF = b"\x0d\x0a"
HEADER_FORMAT = b">%s"
LINE_FORMAT = CRLF + b"%016X:%02X->%02X"
OFFSET_ADJUSTMENT = 0xC00
class ByteDiffException(Exception):
""" Base class for all exceptions """
pass
class LengthMismatchException(ByteDiffException):
""" Throwed when length of input sequences do not match """
pass
2019-02-06 13:04:08 +00:00
class DiffLimitException(ByteDiffException):
""" Throwed when difference limit hit """
pass
2019-02-05 19:05:17 +00:00
def check_positive_int(value):
value = int(value)
if value <= 0:
raise argparse.ArgumentTypeError(
"%s is an invalid positive number value" % value)
return value
def parse_args():
parser = argparse.ArgumentParser(
description="Make .1337 patch file from original and patched files")
parser.add_argument("orig_file",
metavar="ORIG_FILE",
help="original file")
parser.add_argument("patched_file",
metavar="PATCHED_FILE",
help="patched file")
parser.add_argument("output_file",
metavar="OUTPUT_FILE",
nargs='?',
help="filename for patch output. Default: basename of "
"original filename with .1337 extension")
parser.add_argument("-f", "--header-filename",
metavar="FILENAME",
help="filename specified in resulting patch header. "
"Default: basename of original filename.")
parser.add_argument("-l", "--limit",
help="stop after number of differences",
type=check_positive_int)
args = parser.parse_args()
return args
def feed_chunks(f, chunk_size=4096):
""" Reads file-like object with chunks having up to `chunk_size` length """
while True:
buf = f.read(chunk_size)
if not buf:
break
yield buf
2019-02-06 13:22:47 +00:00
def zip_files_bytes(left, right):
2019-02-05 19:05:17 +00:00
""" Iterate over two files, returning pair of bytes.
Throw LengthMismatch if file sizes is uneven. """
class EndMarker(object):
pass
end_marker = EndMarker()
2019-02-06 13:22:47 +00:00
left_iter = itertools.chain.from_iterable(feed_chunks(left))
right_iter = itertools.chain.from_iterable(feed_chunks(right))
for a, b in itertools.zip_longest(left_iter,
right_iter,
fillvalue=end_marker):
if a is end_marker or b is end_marker:
2019-02-05 19:05:17 +00:00
raise LengthMismatchException("Length of input files inequal.")
2019-02-06 13:22:47 +00:00
yield a, b
2019-02-05 19:05:17 +00:00
2019-02-06 13:04:08 +00:00
def diff(left, right, limit=None):
2019-02-06 13:22:47 +00:00
offset = 0
2019-02-06 13:04:08 +00:00
diff_count = 0
2019-02-06 13:22:47 +00:00
for a, b in zip_files_bytes(left, right):
2019-02-05 19:05:17 +00:00
if a != b:
2019-02-06 13:04:08 +00:00
diff_count += 1
if limit is not None and diff_count > limit:
raise DiffLimitException()
2019-02-05 19:05:17 +00:00
yield offset, a, b
2019-02-06 13:22:47 +00:00
offset += 1
2019-02-05 19:05:17 +00:00
def compose_diff_file(orig, patched, output, header, *, limit=None, offset_adjustment=True):
2019-02-05 19:05:17 +00:00
output.write(HEADER_FORMAT % (header.encode('latin-1'),))
2019-02-06 13:04:08 +00:00
adj = OFFSET_ADJUSTMENT if offset_adjustment else 0
for offset, a, b in diff(orig, patched, limit):
output.write(LINE_FORMAT % (offset + adj, a, b))
2019-02-05 19:05:17 +00:00
def main():
args = parse_args()
output_filename = args.output_file
if not output_filename:
orig_bname = os.path.basename(args.orig_file)
before, _, after = orig_bname.rpartition('.')
orig_bname_noext = before if before else after
output_filename = orig_bname_noext + PATCH_EXT
header_filename = args.header_filename
if not header_filename:
header_filename = os.path.basename(args.orig_file)
with open(args.orig_file, 'rb') as orig,\
open(args.patched_file, 'rb') as patched,\
open(output_filename, 'wb') as output:
try:
compose_diff_file(orig, patched, output, header_filename,
limit=args.limit)
2019-02-05 19:05:17 +00:00
except LengthMismatchException:
print("Input files have inequal length. Aborting...",
file=sys.stderr)
2019-02-06 13:04:08 +00:00
except DiffLimitException:
print("Differences limit hit. Aborting...",
file=sys.stderr)
2019-02-05 19:05:17 +00:00
if __name__ == '__main__':
main()