xref: /openbmc/qemu/tests/guest-debug/run-test.py (revision 831734cc)
1#!/usr/bin/env python3
2#
3# Run a gdbstub test case
4#
5# Copyright (c) 2019 Linaro
6#
7# Author: Alex Bennée <alex.bennee@linaro.org>
8#
9# This work is licensed under the terms of the GNU GPL, version 2 or later.
10# See the COPYING file in the top-level directory.
11#
12# SPDX-License-Identifier: GPL-2.0-or-later
13
14import argparse
15import subprocess
16import shutil
17import shlex
18import os
19from time import sleep
20from tempfile import TemporaryDirectory
21
22def get_args():
23    parser = argparse.ArgumentParser(description="A gdbstub test runner")
24    parser.add_argument("--qemu", help="Qemu binary for test",
25                        required=True)
26    parser.add_argument("--qargs", help="Qemu arguments for test")
27    parser.add_argument("--binary", help="Binary to debug",
28                        required=True)
29    parser.add_argument("--test", help="GDB test script",
30                        required=True)
31    parser.add_argument("--gdb", help="The gdb binary to use",
32                        default=None)
33    parser.add_argument("--output", help="A file to redirect output to")
34
35    return parser.parse_args()
36
37
38def log(output, msg):
39    if output:
40        output.write(msg + "\n")
41        output.flush()
42    else:
43        print(msg)
44
45
46if __name__ == '__main__':
47    args = get_args()
48
49    # Search for a gdb we can use
50    if not args.gdb:
51        args.gdb = shutil.which("gdb-multiarch")
52    if not args.gdb:
53        args.gdb = shutil.which("gdb")
54    if not args.gdb:
55        print("We need gdb to run the test")
56        exit(-1)
57    if args.output:
58        output = open(args.output, "w")
59    else:
60        output = None
61
62    socket_dir = TemporaryDirectory("qemu-gdbstub")
63    socket_name = os.path.join(socket_dir.name, "gdbstub.socket")
64
65    # Launch QEMU with binary
66    if "system" in args.qemu:
67        cmd = "%s %s %s -gdb unix:path=%s,server" % (args.qemu,
68                                                     args.qargs,
69                                                     args.binary,
70                                                     socket_name)
71    else:
72        cmd = "%s %s -g %s %s" % (args.qemu, args.qargs, socket_name,
73                                  args.binary)
74
75    log(output, "QEMU CMD: %s" % (cmd))
76    inferior = subprocess.Popen(shlex.split(cmd))
77
78    # Now launch gdb with our test and collect the result
79    gdb_cmd = "%s %s" % (args.gdb, args.binary)
80    # run quietly and ignore .gdbinit
81    gdb_cmd += " -q -n -batch"
82    # disable prompts in case of crash
83    gdb_cmd += " -ex 'set confirm off'"
84    # connect to remote
85    gdb_cmd += " -ex 'target remote %s'" % (socket_name)
86    # finally the test script itself
87    gdb_cmd += " -x %s" % (args.test)
88
89
90    sleep(1)
91    log(output, "GDB CMD: %s" % (gdb_cmd))
92
93    result = subprocess.call(gdb_cmd, shell=True, stdout=output)
94
95    # A negative result is the result of an internal gdb failure like
96    # a crash. We force a return of 0 so we don't fail the test on
97    # account of broken external tools.
98    if result < 0:
99        print("GDB crashed? SKIPPING")
100        exit(0)
101
102    try:
103        inferior.wait(2)
104    except subprocess.TimeoutExpired:
105        print("GDB never connected? Killed guest")
106        inferior.kill()
107
108    exit(result)
109