# Copyright (C) 2015 The Android Open Source Project # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. import math import os import random import shutil import subprocess import sys import tempfile blocksize = 4096 roots = 2 def corrupt(image, offset, length): print "corrupting %d bytes at offset %d" % (length, offset) f = os.open(image, os.O_WRONLY) os.lseek(f, offset, os.SEEK_SET) os.write(f, os.urandom(length)) os.close(f) def corruptmax(image, roots): size = os.stat(image).st_size blocks = int(math.ceil(float(size) / blocksize)) rounds = int(math.ceil(float(blocks) / (255 - roots))) max_errors = int(math.floor(rounds * roots / 2)) * blocksize offset = random.randrange(0, size - max_errors) corrupt(image, offset, max_errors) def encode(image, fec, roots): if subprocess.call([ "fec", "--roots= " + str(roots), image, fec ]) != 0: raise Exception("encoding failed") def decode(image, fec, output): return subprocess.call([ "fec", "--decode", image, fec, output ]) def compare(a, b): return subprocess.call([ "cmp", "-s", a, b ]) def simg2img(image, output): print "creating a non-sparse copy of '%s' to '%s'" % (image, output) if subprocess.call([ "simg2img", image, output]) != 0: raise Exception("simg2img failed") def main(argv): image = argv[0] temp_img = tempfile.NamedTemporaryFile() temp_cor = tempfile.NamedTemporaryFile() temp_fec = tempfile.NamedTemporaryFile() temp_out = tempfile.NamedTemporaryFile() simg2img(image, temp_img.name) simg2img(image, temp_cor.name) encode(image, temp_fec.name, roots) corruptmax(temp_cor.name, roots) if decode(temp_cor.name, temp_fec.name, temp_out.name) != 0: raise Exception("FAILED: failed to correct maximum expected errors") if compare(temp_img.name, temp_out.name) != 0: raise Exception("FAILED: corrected file not identical") else: print "corrected content matches original" corrupt(temp_cor.name, 0, blocksize) if decode(temp_cor.name, temp_fec.name, temp_out.name) == 0: raise Exception("FAILED: corrected more than maximum number of errors?") print "PASSED" if __name__ == '__main__': main(sys.argv[1:])