| #! /usr/bin/env python3 |
| |
| # Copyright (c) 2010 Advanced Micro Devices, Inc. |
| # All rights reserved. |
| # |
| # Redistribution and use in source and binary forms, with or without |
| # modification, are permitted provided that the following conditions are |
| # met: redistributions of source code must retain the above copyright |
| # notice, this list of conditions and the following disclaimer; |
| # redistributions in binary form must reproduce the above copyright |
| # notice, this list of conditions and the following disclaimer in the |
| # documentation and/or other materials provided with the distribution; |
| # neither the name of the copyright holders nor the names of its |
| # contributors may be used to endorse or promote products derived from |
| # this software without specific prior written permission. |
| # |
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| # LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| # A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
| # OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| # SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| # LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| # DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| # THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| # |
| # Author: Steve Reinhardt |
| # |
| |
| # Basic test script for checkpointing. |
| # |
| # Given an M5 command and an interval (in ticks), this script will: |
| # 1. Run the command, dumping periodic checkpoints at the given interval. |
| # 2. Rerun the command for each pair of adjacent checkpoints: |
| # a. Restore from checkpoint N |
| # b. Run until the timestamp of checkpoint N+1 |
| # c. Dump a checkpoint and end the simulation |
| # d. Diff the new checkpoint with the original checkpoint N+1 |
| # |
| # Note that '--' must be used to separate the script args from the |
| # M5 command line. |
| # |
| # Caveats: |
| # |
| # - This script relies on the checkpoint options implemented in |
| # configs/common/Simulation.py, so it works with commands based on |
| # the se.py and fs.py scripts in configs/example, but does not work |
| # directly with the existing regression tests. |
| # - Interleaving simulator and program output can cause discrepancies |
| # in the file position checkpoint information since different runs |
| # have different amount of simulator output. |
| # - Probably lots more issues we don't even know about yet. |
| # |
| # Examples: |
| # |
| # util/checkpoint-tester.py -i 400000 -- build/<ISA>/gem5.opt \ |
| # configs/example/se.py -c tests/test-progs/hello/bin/<isa>/linux/hello \ |
| # --output=progout --errout=progerr |
| # |
| # util/checkpoint-tester.py -i 200000000000 -- build/<ISA>/gem5.opt \ |
| # configs/example/fs.py --script configs/boot/halt.sh |
| # |
| |
| |
| import os, sys, re |
| import subprocess |
| import argparse |
| |
| parser = argparse.ArgumentParser() |
| |
| parser.add_argument('-i', '--interval', type=int) |
| parser.add_argument('-d', '--directory', default='checkpoint-test') |
| parser.add_argument('cmdline', nargs='+', help='gem5 command line') |
| |
| args = parser.parse_args() |
| |
| interval = args.interval |
| |
| if os.path.exists(args.directory): |
| print('Error: test directory', args.directory, 'exists') |
| print(' Tester needs to create directory from scratch') |
| sys.exit(1) |
| |
| top_dir = args.directory |
| os.mkdir(top_dir) |
| |
| cmd_echo = open(os.path.join(top_dir, 'command'), 'w') |
| print(' '.join(sys.argv), file=cmd_echo) |
| cmd_echo.close() |
| |
| m5_binary = args.cmdline[0] |
| |
| args = args.cmdline[1:] |
| |
| checkpoint_args = ['--take-checkpoints', '%d,%d' % (interval, interval)] |
| |
| cptdir = os.path.join(top_dir, 'm5out') |
| |
| print('===> Running initial simulation.') |
| subprocess.call([m5_binary] + ['-red', cptdir] + args + checkpoint_args) |
| |
| dirs = os.listdir(cptdir) |
| expr = re.compile('cpt\.([0-9]*)') |
| cpts = [] |
| for dir in dirs: |
| match = expr.match(dir) |
| if match: |
| cpts.append(int(match.group(1))) |
| |
| cpts.sort() |
| |
| # We test by loading checkpoint N, simulating to (and dumping at) |
| # checkpoint N+1, then comparing the resulting checkpoint with the |
| # original checkpoint N+1. Thus the number of tests we can run is one |
| # less than tha number of checkpoints. |
| for i in range(1, len(cpts)): |
| print('===> Running test %d of %d.' % (i, len(cpts)-1)) |
| checkpoint_args = ['--take-checkpoints', '%d,%d' % (cpts[i], interval)] |
| mydir = os.path.join(top_dir, 'test.%d' % i) |
| subprocess.call([m5_binary] + ['-red', mydir] + args + checkpoint_args + |
| ['--max-checkpoints' , '1', '--checkpoint-dir', cptdir, |
| '--checkpoint-restore', str(i)]) |
| cpt_name = 'cpt.%d' % cpts[i] |
| diff_name = os.path.join(mydir, 'diffout') |
| diffout = open(diff_name, 'w') |
| subprocess.call(['diff', '-ru', '-I', '^##.*', |
| '%s/%s' % (cptdir, cpt_name), |
| '%s/%s' % (mydir, cpt_name)], stdout=diffout) |
| diffout.close() |
| # print out the diff |
| diffout = open(diff_name) |
| print(diffout.read(), end=' ') |
| diffout.close() |
| |
| |