2020-03-16 20:21:49 +03:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
#
|
|
|
|
# Run a gdbstub test case
|
|
|
|
#
|
|
|
|
# Copyright (c) 2019 Linaro
|
|
|
|
#
|
|
|
|
# Author: Alex Bennée <alex.bennee@linaro.org>
|
|
|
|
#
|
|
|
|
# This work is licensed under the terms of the GNU GPL, version 2 or later.
|
|
|
|
# See the COPYING file in the top-level directory.
|
|
|
|
#
|
|
|
|
# SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
|
|
|
|
import argparse
|
|
|
|
import subprocess
|
|
|
|
import shutil
|
|
|
|
import shlex
|
2020-04-30 22:01:20 +03:00
|
|
|
import os
|
2021-01-09 01:42:41 +03:00
|
|
|
from time import sleep
|
2020-04-30 22:01:20 +03:00
|
|
|
from tempfile import TemporaryDirectory
|
2020-03-16 20:21:49 +03:00
|
|
|
|
|
|
|
def get_args():
|
|
|
|
parser = argparse.ArgumentParser(description="A gdbstub test runner")
|
|
|
|
parser.add_argument("--qemu", help="Qemu binary for test",
|
|
|
|
required=True)
|
|
|
|
parser.add_argument("--qargs", help="Qemu arguments for test")
|
|
|
|
parser.add_argument("--binary", help="Binary to debug",
|
|
|
|
required=True)
|
2023-05-04 18:37:32 +03:00
|
|
|
parser.add_argument("--test", help="GDB test script")
|
2021-01-09 01:42:41 +03:00
|
|
|
parser.add_argument("--gdb", help="The gdb binary to use",
|
|
|
|
default=None)
|
2023-05-04 18:37:32 +03:00
|
|
|
parser.add_argument("--gdb-args", help="Additional gdb arguments")
|
2021-01-09 01:42:41 +03:00
|
|
|
parser.add_argument("--output", help="A file to redirect output to")
|
2023-05-04 18:37:32 +03:00
|
|
|
parser.add_argument("--stderr", help="A file to redirect stderr to")
|
2020-03-16 20:21:49 +03:00
|
|
|
|
|
|
|
return parser.parse_args()
|
|
|
|
|
2021-01-09 01:42:41 +03:00
|
|
|
|
|
|
|
def log(output, msg):
|
|
|
|
if output:
|
|
|
|
output.write(msg + "\n")
|
|
|
|
output.flush()
|
|
|
|
else:
|
|
|
|
print(msg)
|
|
|
|
|
|
|
|
|
2020-03-16 20:21:49 +03:00
|
|
|
if __name__ == '__main__':
|
|
|
|
args = get_args()
|
|
|
|
|
|
|
|
# Search for a gdb we can use
|
|
|
|
if not args.gdb:
|
|
|
|
args.gdb = shutil.which("gdb-multiarch")
|
|
|
|
if not args.gdb:
|
|
|
|
args.gdb = shutil.which("gdb")
|
|
|
|
if not args.gdb:
|
|
|
|
print("We need gdb to run the test")
|
|
|
|
exit(-1)
|
2021-01-09 01:42:41 +03:00
|
|
|
if args.output:
|
|
|
|
output = open(args.output, "w")
|
|
|
|
else:
|
|
|
|
output = None
|
2023-05-04 18:37:32 +03:00
|
|
|
if args.stderr:
|
|
|
|
stderr = open(args.stderr, "w")
|
|
|
|
else:
|
|
|
|
stderr = None
|
2020-03-16 20:21:49 +03:00
|
|
|
|
2020-04-30 22:01:20 +03:00
|
|
|
socket_dir = TemporaryDirectory("qemu-gdbstub")
|
|
|
|
socket_name = os.path.join(socket_dir.name, "gdbstub.socket")
|
|
|
|
|
2020-03-16 20:21:49 +03:00
|
|
|
# Launch QEMU with binary
|
|
|
|
if "system" in args.qemu:
|
2023-08-10 18:36:38 +03:00
|
|
|
cmd = f'{args.qemu} {args.qargs} {args.binary}' \
|
|
|
|
f' -S -gdb unix:path={socket_name},server=on'
|
2020-03-16 20:21:49 +03:00
|
|
|
else:
|
2023-08-10 18:36:38 +03:00
|
|
|
cmd = f'{args.qemu} {args.qargs} -g {socket_name} {args.binary}'
|
2020-03-16 20:21:49 +03:00
|
|
|
|
2021-01-09 01:42:41 +03:00
|
|
|
log(output, "QEMU CMD: %s" % (cmd))
|
2020-03-16 20:21:49 +03:00
|
|
|
inferior = subprocess.Popen(shlex.split(cmd))
|
|
|
|
|
|
|
|
# Now launch gdb with our test and collect the result
|
2020-04-30 22:01:16 +03:00
|
|
|
gdb_cmd = "%s %s" % (args.gdb, args.binary)
|
2023-05-04 18:37:32 +03:00
|
|
|
if args.gdb_args:
|
|
|
|
gdb_cmd += " %s" % (args.gdb_args)
|
2020-04-30 22:01:16 +03:00
|
|
|
# run quietly and ignore .gdbinit
|
|
|
|
gdb_cmd += " -q -n -batch"
|
2023-08-29 19:15:23 +03:00
|
|
|
# disable pagination
|
|
|
|
gdb_cmd += " -ex 'set pagination off'"
|
2020-04-30 22:01:16 +03:00
|
|
|
# disable prompts in case of crash
|
|
|
|
gdb_cmd += " -ex 'set confirm off'"
|
|
|
|
# connect to remote
|
2021-01-09 01:42:41 +03:00
|
|
|
gdb_cmd += " -ex 'target remote %s'" % (socket_name)
|
2020-04-30 22:01:16 +03:00
|
|
|
# finally the test script itself
|
2023-05-04 18:37:32 +03:00
|
|
|
if args.test:
|
|
|
|
gdb_cmd += " -x %s" % (args.test)
|
2020-04-30 22:01:16 +03:00
|
|
|
|
2020-03-16 20:21:49 +03:00
|
|
|
|
2021-01-09 01:42:41 +03:00
|
|
|
sleep(1)
|
|
|
|
log(output, "GDB CMD: %s" % (gdb_cmd))
|
|
|
|
|
2024-01-29 12:32:15 +03:00
|
|
|
gdb_env = dict(os.environ)
|
|
|
|
gdb_pythonpath = gdb_env.get("PYTHONPATH", "").split(os.pathsep)
|
|
|
|
gdb_pythonpath.append(os.path.dirname(os.path.realpath(__file__)))
|
|
|
|
gdb_env["PYTHONPATH"] = os.pathsep.join(gdb_pythonpath)
|
|
|
|
result = subprocess.call(gdb_cmd, shell=True, stdout=output, stderr=stderr,
|
|
|
|
env=gdb_env)
|
2020-03-16 20:21:49 +03:00
|
|
|
|
2022-04-19 12:10:20 +03:00
|
|
|
# A result of greater than 128 indicates a fatal signal (likely a
|
|
|
|
# crash due to gdb internal failure). That's a problem for GDB and
|
|
|
|
# not the test so we force a return of 0 so we don't fail the test on
|
2020-04-30 22:01:16 +03:00
|
|
|
# account of broken external tools.
|
2022-04-19 12:10:20 +03:00
|
|
|
if result > 128:
|
|
|
|
log(output, "GDB crashed? (%d, %d) SKIPPING" % (result, result - 128))
|
2020-04-30 22:01:16 +03:00
|
|
|
exit(0)
|
|
|
|
|
2020-05-13 20:51:25 +03:00
|
|
|
try:
|
|
|
|
inferior.wait(2)
|
|
|
|
except subprocess.TimeoutExpired:
|
2022-04-19 12:10:20 +03:00
|
|
|
log(output, "GDB never connected? Killed guest")
|
2020-05-13 20:51:25 +03:00
|
|
|
inferior.kill()
|
|
|
|
|
2020-03-16 20:21:49 +03:00
|
|
|
exit(result)
|