| from __future__ import print_function |
| # |
| # Test some of the system debug features with the multiarch memory |
| # test. It is a port of the original vmlinux focused test case but |
| # using the "memory" test instead. |
| # |
| # This is launched via tests/guest-debug/run-test.py |
| # |
| |
| import gdb |
| import sys |
| |
| failcount = 0 |
| |
| |
| def report(cond, msg): |
| "Report success/fail of test" |
| if cond: |
| print("PASS: %s" % (msg)) |
| else: |
| print("FAIL: %s" % (msg)) |
| global failcount |
| failcount += 1 |
| |
| |
| def check_step(): |
| "Step an instruction, check it moved." |
| start_pc = gdb.parse_and_eval('$pc') |
| gdb.execute("si") |
| end_pc = gdb.parse_and_eval('$pc') |
| |
| return not (start_pc == end_pc) |
| |
| |
| # |
| # Currently it's hard to create a hbreak with the pure python API and |
| # manually matching PC to symbol address is a bit flaky thanks to |
| # function prologues. However internally QEMU's gdbstub treats them |
| # the same as normal breakpoints so it will do for now. |
| # |
| def check_break(sym_name): |
| "Setup breakpoint, continue and check we stopped." |
| sym, ok = gdb.lookup_symbol(sym_name) |
| bp = gdb.Breakpoint(sym_name, gdb.BP_BREAKPOINT) |
| |
| gdb.execute("c") |
| |
| # hopefully we came back |
| end_pc = gdb.parse_and_eval('$pc') |
| report(bp.hit_count == 1, |
| "break @ %s (%s %d hits)" % (end_pc, sym.value(), bp.hit_count)) |
| |
| bp.delete() |
| |
| |
| def do_one_watch(sym, wtype, text): |
| |
| wp = gdb.Breakpoint(sym, gdb.BP_WATCHPOINT, wtype) |
| gdb.execute("c") |
| report_str = "%s for %s" % (text, sym) |
| |
| if wp.hit_count > 0: |
| report(True, report_str) |
| wp.delete() |
| else: |
| report(False, report_str) |
| |
| |
| def check_watches(sym_name): |
| "Watch a symbol for any access." |
| |
| # Should hit for any read |
| do_one_watch(sym_name, gdb.WP_ACCESS, "awatch") |
| |
| # Again should hit for reads |
| do_one_watch(sym_name, gdb.WP_READ, "rwatch") |
| |
| # Finally when it is written |
| do_one_watch(sym_name, gdb.WP_WRITE, "watch") |
| |
| |
| def run_test(): |
| "Run through the tests one by one" |
| |
| print("Checking we can step the first few instructions") |
| step_ok = 0 |
| for i in range(3): |
| if check_step(): |
| step_ok += 1 |
| |
| report(step_ok == 3, "single step in boot code") |
| |
| # If we get here we have missed some of the other breakpoints. |
| print("Setup catch-all for _exit") |
| cbp = gdb.Breakpoint("_exit", gdb.BP_BREAKPOINT) |
| |
| check_break("main") |
| check_watches("test_data[128]") |
| |
| report(cbp.hit_count == 0, "didn't reach backstop") |
| |
| # |
| # This runs as the script it sourced (via -x, via run-test.py) |
| # |
| try: |
| inferior = gdb.selected_inferior() |
| arch = inferior.architecture() |
| print("ATTACHED: %s" % arch.name()) |
| except (gdb.error, AttributeError): |
| print("SKIPPING (not connected)", file=sys.stderr) |
| exit(0) |
| |
| if gdb.parse_and_eval('$pc') == 0: |
| print("SKIP: PC not set") |
| exit(0) |
| |
| try: |
| # Run the actual tests |
| run_test() |
| except (gdb.error): |
| print("GDB Exception: %s" % (sys.exc_info()[0])) |
| failcount += 1 |
| pass |
| |
| # Finally kill the inferior and exit gdb with a count of failures |
| gdb.execute("kill") |
| exit(failcount) |