* [PATCH v3 0/3] linux-user: Allow gdbstub to ignore page protection
@ 2024-01-16 0:31 Ilya Leoshkevich
2024-01-16 0:31 ` [PATCH v3 1/3] " Ilya Leoshkevich
` (2 more replies)
0 siblings, 3 replies; 11+ messages in thread
From: Ilya Leoshkevich @ 2024-01-16 0:31 UTC (permalink / raw)
To: Richard Henderson, Peter Maydell, Alex Bennée,
David Hildenbrand
Cc: Paolo Bonzini, Philippe Mathieu-Daudé, qemu-devel, qemu-arm,
qemu-s390x, Ilya Leoshkevich
v2: https://lists.gnu.org/archive/html/qemu-devel/2024-01/msg01592.html
v2 -> v3: Add Richard's R-b on [1/3].
Fix printing the architecture name and the number of failures
in test_gdbstub.py.
Patches that need review: [2/3] and [3/3].
v1: https://lists.gnu.org/archive/html/qemu-devel/2024-01/msg01314.html
v1 -> v2: Use /proc/self/mem as a fallback. Handle TB invalidation
(Richard).
Test cross-page accesses.
RFC: https://lists.gnu.org/archive/html/qemu-devel/2023-12/msg02044.html
RFC -> v1: Use /proc/self/mem and accept that this will not work
without /proc.
Factor out a couple functions for gdbstub testing.
Add a test.
Hi,
I've noticed that gdbstub behaves differently from gdbserver in that it
doesn't allow reading non-readable pages. This series improves the
situation by using the same mechanism as gdbserver: /proc/self/mem.
Best regards,
Ilya
Ilya Leoshkevich (3):
linux-user: Allow gdbstub to ignore page protection
tests/tcg: Factor out gdbstub test functions
tests/tcg: Add the PROT_NONE gdbstub test
cpu-target.c | 76 +++++++++++++++----
tests/guest-debug/run-test.py | 7 +-
tests/guest-debug/test_gdbstub.py | 58 ++++++++++++++
tests/tcg/aarch64/gdbstub/test-sve-ioctl.py | 34 +--------
tests/tcg/aarch64/gdbstub/test-sve.py | 33 +-------
tests/tcg/multiarch/Makefile.target | 9 ++-
tests/tcg/multiarch/gdbstub/interrupt.py | 47 ++----------
tests/tcg/multiarch/gdbstub/memory.py | 41 +---------
tests/tcg/multiarch/gdbstub/prot-none.py | 22 ++++++
tests/tcg/multiarch/gdbstub/registers.py | 41 ++--------
tests/tcg/multiarch/gdbstub/sha1.py | 40 ++--------
.../multiarch/gdbstub/test-proc-mappings.py | 39 +---------
.../multiarch/gdbstub/test-qxfer-auxv-read.py | 37 +--------
.../gdbstub/test-thread-breakpoint.py | 37 +--------
tests/tcg/multiarch/prot-none.c | 40 ++++++++++
tests/tcg/s390x/gdbstub/test-signals-s390x.py | 42 +---------
tests/tcg/s390x/gdbstub/test-svc.py | 39 +---------
17 files changed, 229 insertions(+), 413 deletions(-)
create mode 100644 tests/guest-debug/test_gdbstub.py
create mode 100644 tests/tcg/multiarch/gdbstub/prot-none.py
create mode 100644 tests/tcg/multiarch/prot-none.c
--
2.43.0
^ permalink raw reply [flat|nested] 11+ messages in thread
* [PATCH v3 1/3] linux-user: Allow gdbstub to ignore page protection
2024-01-16 0:31 [PATCH v3 0/3] linux-user: Allow gdbstub to ignore page protection Ilya Leoshkevich
@ 2024-01-16 0:31 ` Ilya Leoshkevich
2024-01-16 0:31 ` [PATCH v3 2/3] tests/tcg: Factor out gdbstub test functions Ilya Leoshkevich
2024-01-16 0:31 ` [PATCH v3 3/3] tests/tcg: Add the PROT_NONE gdbstub test Ilya Leoshkevich
2 siblings, 0 replies; 11+ messages in thread
From: Ilya Leoshkevich @ 2024-01-16 0:31 UTC (permalink / raw)
To: Richard Henderson, Peter Maydell, Alex Bennée,
David Hildenbrand
Cc: Paolo Bonzini, Philippe Mathieu-Daudé, qemu-devel, qemu-arm,
qemu-s390x, Ilya Leoshkevich
gdbserver ignores page protection by virtue of using /proc/$pid/mem.
Teach qemu gdbstub to do this too. This will not work if /proc is not
mounted; accept this limitation.
One alternative is to temporarily grant the missing PROT_* bit, but
this is inherently racy. Another alternative is self-debugging with
ptrace(POKE), which will break if QEMU itself is being debugged - a
much more severe limitation.
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
Signed-off-by: Ilya Leoshkevich <iii@linux.ibm.com>
---
cpu-target.c | 76 +++++++++++++++++++++++++++++++++++++++++-----------
1 file changed, 61 insertions(+), 15 deletions(-)
diff --git a/cpu-target.c b/cpu-target.c
index 5eecd7ea2d7..723f6af5fba 100644
--- a/cpu-target.c
+++ b/cpu-target.c
@@ -406,6 +406,9 @@ int cpu_memory_rw_debug(CPUState *cpu, vaddr addr,
vaddr l, page;
void * p;
uint8_t *buf = ptr;
+ ssize_t written;
+ int ret = -1;
+ int fd = -1;
while (len > 0) {
page = addr & TARGET_PAGE_MASK;
@@ -413,30 +416,73 @@ int cpu_memory_rw_debug(CPUState *cpu, vaddr addr,
if (l > len)
l = len;
flags = page_get_flags(page);
- if (!(flags & PAGE_VALID))
- return -1;
+ if (!(flags & PAGE_VALID)) {
+ goto out_close;
+ }
if (is_write) {
- if (!(flags & PAGE_WRITE))
- return -1;
- /* XXX: this code should not depend on lock_user */
- if (!(p = lock_user(VERIFY_WRITE, addr, l, 0)))
- return -1;
- memcpy(p, buf, l);
- unlock_user(p, addr, l);
- } else {
- if (!(flags & PAGE_READ))
- return -1;
+ if (flags & PAGE_WRITE) {
+ /* XXX: this code should not depend on lock_user */
+ p = lock_user(VERIFY_WRITE, addr, l, 0);
+ if (!p) {
+ goto out_close;
+ }
+ memcpy(p, buf, l);
+ unlock_user(p, addr, l);
+ } else {
+ /* Bypass the host page protection using ptrace. */
+ if (fd == -1) {
+ fd = open("/proc/self/mem", O_WRONLY);
+ if (fd == -1) {
+ goto out;
+ }
+ }
+ /*
+ * If there is a TranslationBlock and we weren't bypassing the
+ * host page protection, the memcpy() above would SEGV,
+ * ultimately leading to page_unprotect(). So invalidate the
+ * translations manually. Both invalidation and pwrite() must
+ * be under mmap_lock() in order to prevent the creation of
+ * another TranslationBlock in between.
+ */
+ mmap_lock();
+ tb_invalidate_phys_range(addr, addr + l - 1);
+ written = pwrite(fd, buf, l, (off_t)g2h_untagged(addr));
+ mmap_unlock();
+ if (written != l) {
+ goto out_close;
+ }
+ }
+ } else if (flags & PAGE_READ) {
/* XXX: this code should not depend on lock_user */
- if (!(p = lock_user(VERIFY_READ, addr, l, 1)))
- return -1;
+ p = lock_user(VERIFY_READ, addr, l, 1);
+ if (!p) {
+ goto out_close;
+ }
memcpy(buf, p, l);
unlock_user(p, addr, 0);
+ } else {
+ /* Bypass the host page protection using ptrace. */
+ if (fd == -1) {
+ fd = open("/proc/self/mem", O_RDONLY);
+ if (fd == -1) {
+ goto out;
+ }
+ }
+ if (pread(fd, buf, l, (off_t)g2h_untagged(addr)) != l) {
+ goto out_close;
+ }
}
len -= l;
buf += l;
addr += l;
}
- return 0;
+ ret = 0;
+out_close:
+ if (fd != -1) {
+ close(fd);
+ }
+out:
+ return ret;
}
#endif
--
2.43.0
^ permalink raw reply related [flat|nested] 11+ messages in thread
* [PATCH v3 2/3] tests/tcg: Factor out gdbstub test functions
2024-01-16 0:31 [PATCH v3 0/3] linux-user: Allow gdbstub to ignore page protection Ilya Leoshkevich
2024-01-16 0:31 ` [PATCH v3 1/3] " Ilya Leoshkevich
@ 2024-01-16 0:31 ` Ilya Leoshkevich
2024-01-22 16:00 ` Alex Bennée
2024-01-16 0:31 ` [PATCH v3 3/3] tests/tcg: Add the PROT_NONE gdbstub test Ilya Leoshkevich
2 siblings, 1 reply; 11+ messages in thread
From: Ilya Leoshkevich @ 2024-01-16 0:31 UTC (permalink / raw)
To: Richard Henderson, Peter Maydell, Alex Bennée,
David Hildenbrand
Cc: Paolo Bonzini, Philippe Mathieu-Daudé, qemu-devel, qemu-arm,
qemu-s390x, Ilya Leoshkevich
Both the report() function as well as the initial gdbstub test sequence
are copy-pasted into ~10 files with slight modifications. This
indicates that they are indeed generic, so factor them out. While
at it, add a few newlines to make the formatting closer to PEP-8.
Signed-off-by: Ilya Leoshkevich <iii@linux.ibm.com>
---
tests/guest-debug/run-test.py | 7 ++-
tests/guest-debug/test_gdbstub.py | 58 +++++++++++++++++++
tests/tcg/aarch64/gdbstub/test-sve-ioctl.py | 34 +----------
tests/tcg/aarch64/gdbstub/test-sve.py | 33 +----------
tests/tcg/multiarch/gdbstub/interrupt.py | 47 ++-------------
tests/tcg/multiarch/gdbstub/memory.py | 41 +------------
tests/tcg/multiarch/gdbstub/registers.py | 41 ++-----------
tests/tcg/multiarch/gdbstub/sha1.py | 40 ++-----------
.../multiarch/gdbstub/test-proc-mappings.py | 39 +------------
.../multiarch/gdbstub/test-qxfer-auxv-read.py | 37 +-----------
| 37 +-----------
tests/tcg/s390x/gdbstub/test-signals-s390x.py | 42 +-------------
tests/tcg/s390x/gdbstub/test-svc.py | 39 +------------
13 files changed, 98 insertions(+), 397 deletions(-)
create mode 100644 tests/guest-debug/test_gdbstub.py
diff --git a/tests/guest-debug/run-test.py b/tests/guest-debug/run-test.py
index b13b27d4b19..368ff8a8903 100755
--- a/tests/guest-debug/run-test.py
+++ b/tests/guest-debug/run-test.py
@@ -97,7 +97,12 @@ def log(output, msg):
sleep(1)
log(output, "GDB CMD: %s" % (gdb_cmd))
- result = subprocess.call(gdb_cmd, shell=True, stdout=output, stderr=stderr)
+ 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)
# 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
diff --git a/tests/guest-debug/test_gdbstub.py b/tests/guest-debug/test_gdbstub.py
new file mode 100644
index 00000000000..a71cdaa915a
--- /dev/null
+++ b/tests/guest-debug/test_gdbstub.py
@@ -0,0 +1,58 @@
+"""Helper functions for gdbstub testing
+
+"""
+from __future__ import print_function
+import gdb
+import sys
+import traceback
+
+fail_count = 0
+
+
+def report(cond, msg):
+ """Report success/fail of a test"""
+ if cond:
+ print("PASS: {}".format(msg))
+ else:
+ print("FAIL: {}".format(msg))
+ global fail_count
+ fail_count += 1
+
+
+def main(test, expected_arch=None):
+ """Run a test function
+
+ This runs as the script it sourced (via -x, via run-test.py)."""
+ try:
+ inferior = gdb.selected_inferior()
+ arch = inferior.architecture()
+ print("ATTACHED: {}".format(arch.name()))
+ if expected_arch is not None:
+ report(arch.name() == expected_arch,
+ "connected to {}".format(expected_arch))
+ except (gdb.error, AttributeError):
+ print("SKIP: not connected")
+ exit(0)
+
+ if gdb.parse_and_eval("$pc") == 0:
+ print("SKIP: PC not set")
+ exit(0)
+
+ try:
+ test()
+ except:
+ print("GDB Exception:")
+ traceback.print_exc(file=sys.stdout)
+ global fail_count
+ fail_count += 1
+ import code
+ code.InteractiveConsole(locals=globals()).interact()
+ raise
+
+ try:
+ gdb.execute("kill")
+ except gdb.error:
+ pass
+
+ print("All tests complete: {} failures".format(fail_count))
+ exit(fail_count)
diff --git a/tests/tcg/aarch64/gdbstub/test-sve-ioctl.py b/tests/tcg/aarch64/gdbstub/test-sve-ioctl.py
index ee8d467e59d..a78a3a2514d 100644
--- a/tests/tcg/aarch64/gdbstub/test-sve-ioctl.py
+++ b/tests/tcg/aarch64/gdbstub/test-sve-ioctl.py
@@ -8,19 +8,10 @@
#
import gdb
-import sys
+from test_gdbstub import main, report
initial_vlen = 0
-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
class TestBreakpoint(gdb.Breakpoint):
def __init__(self, sym_name="__sve_ld_done"):
@@ -64,26 +55,5 @@ def run_test():
gdb.execute("c")
-#
-# This runs as the script it sourced (via -x, via run-test.py)
-#
-try:
- inferior = gdb.selected_inferior()
- arch = inferior.architecture()
- report(arch.name() == "aarch64", "connected to aarch64")
-except (gdb.error, AttributeError):
- print("SKIPPING (not connected)", file=sys.stderr)
- exit(0)
-
-try:
- # Run the actual tests
- run_test()
-except:
- print ("GDB Exception: %s" % (sys.exc_info()[0]))
- failcount += 1
- import code
- code.InteractiveConsole(locals=globals()).interact()
- raise
-print("All tests complete: %d failures" % failcount)
-exit(failcount)
+main(run_test, expected_arch="aarch64")
diff --git a/tests/tcg/aarch64/gdbstub/test-sve.py b/tests/tcg/aarch64/gdbstub/test-sve.py
index afd8ece98dd..84cdcd4a32e 100644
--- a/tests/tcg/aarch64/gdbstub/test-sve.py
+++ b/tests/tcg/aarch64/gdbstub/test-sve.py
@@ -6,20 +6,10 @@
#
import gdb
-import sys
+from test_gdbstub import main, report
MAGIC = 0xDEADBEEF
-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 run_test():
"Run through the tests one by one"
@@ -54,24 +44,5 @@ def run_test():
report(str(v.type) == "uint64_t", "size of %s" % (reg))
report(int(v) == MAGIC, "%s is 0x%x" % (reg, MAGIC))
-#
-# This runs as the script it sourced (via -x, via run-test.py)
-#
-try:
- inferior = gdb.selected_inferior()
- arch = inferior.architecture()
- report(arch.name() == "aarch64", "connected to aarch64")
-except (gdb.error, AttributeError):
- print("SKIPPING (not connected)", file=sys.stderr)
- exit(0)
-
-try:
- # Run the actual tests
- run_test()
-except:
- print ("GDB Exception: %s" % (sys.exc_info()[0]))
- failcount += 1
-
-print("All tests complete: %d failures" % failcount)
-exit(failcount)
+main(run_test, expected_arch="aarch64")
diff --git a/tests/tcg/multiarch/gdbstub/interrupt.py b/tests/tcg/multiarch/gdbstub/interrupt.py
index c016e7afbbf..90a45b5140a 100644
--- a/tests/tcg/multiarch/gdbstub/interrupt.py
+++ b/tests/tcg/multiarch/gdbstub/interrupt.py
@@ -8,19 +8,7 @@
#
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
+from test_gdbstub import main, report
def check_interrupt(thread):
@@ -59,6 +47,9 @@ def run_test():
Test if interrupting the code always lands us on the same thread when
running with scheduler-lock enabled.
"""
+ if len(gdb.selected_inferior().threads()) == 1:
+ print("SKIP: set to run on a single thread")
+ exit(0)
gdb.execute("set scheduler-locking on")
for thread in gdb.selected_inferior().threads():
@@ -66,32 +57,4 @@ def run_test():
"thread %d resumes correctly on interrupt" % thread.num)
-#
-# 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)
-if len(gdb.selected_inferior().threads()) == 1:
- print("SKIP: set to run on a single thread")
- 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)
+main(run_test)
diff --git a/tests/tcg/multiarch/gdbstub/memory.py b/tests/tcg/multiarch/gdbstub/memory.py
index fb1d06b7bb7..532b92e7fb3 100644
--- a/tests/tcg/multiarch/gdbstub/memory.py
+++ b/tests/tcg/multiarch/gdbstub/memory.py
@@ -9,18 +9,7 @@
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
+from test_gdbstub import main, report
def check_step():
@@ -99,29 +88,5 @@ def run_test():
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)
+
+main(run_test)
diff --git a/tests/tcg/multiarch/gdbstub/registers.py b/tests/tcg/multiarch/gdbstub/registers.py
index 688c0611072..b3d13cb077f 100644
--- a/tests/tcg/multiarch/gdbstub/registers.py
+++ b/tests/tcg/multiarch/gdbstub/registers.py
@@ -7,20 +7,11 @@
# SPDX-License-Identifier: GPL-2.0-or-later
import gdb
-import sys
import xml.etree.ElementTree as ET
+from test_gdbstub import main, report
-initial_vlen = 0
-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
+initial_vlen = 0
def fetch_xml_regmap():
@@ -75,6 +66,7 @@ def fetch_xml_regmap():
return reg_map
+
def get_register_by_regnum(reg_map, regnum):
"""
Helper to find a register from the map via its XML regnum
@@ -84,6 +76,7 @@ def get_register_by_regnum(reg_map, regnum):
return entry
return None
+
def crosscheck_remote_xml(reg_map):
"""
Cross-check the list of remote-registers with the XML info.
@@ -144,6 +137,7 @@ def crosscheck_remote_xml(reg_map):
elif "seen" not in x_reg:
print(f"{x_reg} wasn't seen in remote-registers")
+
def initial_register_read(reg_map):
"""
Do an initial read of all registers that we know gdb cares about
@@ -214,27 +208,4 @@ def run_test():
complete_and_diff(reg_map)
-#
-# 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_test()
-except (gdb.error):
- print ("GDB Exception: %s" % (sys.exc_info()[0]))
- failcount += 1
- pass
-
-print("All tests complete: %d failures" % failcount)
-exit(failcount)
+main(run_test)
diff --git a/tests/tcg/multiarch/gdbstub/sha1.py b/tests/tcg/multiarch/gdbstub/sha1.py
index 416728415f9..1ce711a402c 100644
--- a/tests/tcg/multiarch/gdbstub/sha1.py
+++ b/tests/tcg/multiarch/gdbstub/sha1.py
@@ -7,19 +7,11 @@
#
import gdb
-import sys
+from test_gdbstub import main, report
+
initial_vlen = 0
-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_break(sym_name):
"Setup breakpoint, continue and check we stopped."
@@ -35,6 +27,7 @@ def check_break(sym_name):
bp.delete()
+
def run_test():
"Run through the tests one by one"
@@ -57,28 +50,5 @@ def run_test():
# finally check we don't barf inspecting registers
gdb.execute("info registers")
-#
-# 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
-
-print("All tests complete: %d failures" % failcount)
-exit(failcount)
+
+main(run_test)
diff --git a/tests/tcg/multiarch/gdbstub/test-proc-mappings.py b/tests/tcg/multiarch/gdbstub/test-proc-mappings.py
index 04ec61d2197..564613fabf0 100644
--- a/tests/tcg/multiarch/gdbstub/test-proc-mappings.py
+++ b/tests/tcg/multiarch/gdbstub/test-proc-mappings.py
@@ -3,20 +3,7 @@
This runs as a sourced script (via -x, via run-test.py)."""
from __future__ import print_function
import gdb
-import sys
-
-
-n_failures = 0
-
-
-def report(cond, msg):
- """Report success/fail of a test"""
- if cond:
- print("PASS: {}".format(msg))
- else:
- print("FAIL: {}".format(msg))
- global n_failures
- n_failures += 1
+from test_gdbstub import main, report
def run_test():
@@ -37,26 +24,4 @@ def run_test():
# report("/sha1" in mappings, "Found the test binary name in the mappings")
-def main():
- """Prepare the environment and run through the tests"""
- try:
- inferior = gdb.selected_inferior()
- print("ATTACHED: {}".format(inferior.architecture().name()))
- except (gdb.error, AttributeError):
- print("SKIPPING (not connected)")
- 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:
- report(False, "GDB Exception: {}".format(sys.exc_info()[0]))
- print("All tests complete: %d failures" % n_failures)
- exit(n_failures)
-
-
-main()
+main(run_test)
diff --git a/tests/tcg/multiarch/gdbstub/test-qxfer-auxv-read.py b/tests/tcg/multiarch/gdbstub/test-qxfer-auxv-read.py
index 926fa962b77..00c26ab4a95 100644
--- a/tests/tcg/multiarch/gdbstub/test-qxfer-auxv-read.py
+++ b/tests/tcg/multiarch/gdbstub/test-qxfer-auxv-read.py
@@ -6,18 +6,8 @@
#
import gdb
-import sys
+from test_gdbstub import main, report
-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 run_test():
"Run through the tests one by one"
@@ -26,28 +16,5 @@ def run_test():
report(isinstance(auxv, str), "Fetched auxv from inferior")
report(auxv.find("sha1"), "Found test binary name in auxv")
-#
-# 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
-print("All tests complete: %d failures" % failcount)
-exit(failcount)
+main(run_test)
--git a/tests/tcg/multiarch/gdbstub/test-thread-breakpoint.py b/tests/tcg/multiarch/gdbstub/test-thread-breakpoint.py
index e57d2a8db8b..4d6b6b9fbe7 100644
--- a/tests/tcg/multiarch/gdbstub/test-thread-breakpoint.py
+++ b/tests/tcg/multiarch/gdbstub/test-thread-breakpoint.py
@@ -6,18 +6,8 @@
#
import gdb
-import sys
+from test_gdbstub import main, report
-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 run_test():
"Run through the tests one by one"
@@ -29,28 +19,5 @@ def run_test():
frame = gdb.selected_frame()
report(str(frame.function()) == "thread1_func", "break @ %s"%frame)
-#
-# 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
-print("All tests complete: %d failures" % failcount)
-exit(failcount)
+main(run_test)
diff --git a/tests/tcg/s390x/gdbstub/test-signals-s390x.py b/tests/tcg/s390x/gdbstub/test-signals-s390x.py
index ca2bbc0b03e..b6b7b39fc46 100644
--- a/tests/tcg/s390x/gdbstub/test-signals-s390x.py
+++ b/tests/tcg/s390x/gdbstub/test-signals-s390x.py
@@ -7,19 +7,7 @@
#
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
+from test_gdbstub import main, report
def run_test():
@@ -42,31 +30,7 @@ def run_test():
gdb.Breakpoint("_exit")
gdb.execute("c")
status = int(gdb.parse_and_eval("$r2"))
- report(status == 0, "status == 0");
-
-
-#
-# 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)
+ report(status == 0, "status == 0")
-try:
- # Run the actual tests
- run_test()
-except (gdb.error):
- print("GDB Exception: %s" % (sys.exc_info()[0]))
- failcount += 1
- pass
-print("All tests complete: %d failures" % failcount)
-exit(failcount)
+main(run_test)
diff --git a/tests/tcg/s390x/gdbstub/test-svc.py b/tests/tcg/s390x/gdbstub/test-svc.py
index 804705fede9..17210b4e020 100644
--- a/tests/tcg/s390x/gdbstub/test-svc.py
+++ b/tests/tcg/s390x/gdbstub/test-svc.py
@@ -3,20 +3,7 @@
This runs as a sourced script (via -x, via run-test.py)."""
from __future__ import print_function
import gdb
-import sys
-
-
-n_failures = 0
-
-
-def report(cond, msg):
- """Report success/fail of a test"""
- if cond:
- print("PASS: {}".format(msg))
- else:
- print("FAIL: {}".format(msg))
- global n_failures
- n_failures += 1
+from test_gdbstub import main, report
def run_test():
@@ -35,26 +22,4 @@ def run_test():
gdb.execute("si")
-def main():
- """Prepare the environment and run through the tests"""
- try:
- inferior = gdb.selected_inferior()
- print("ATTACHED: {}".format(inferior.architecture().name()))
- except (gdb.error, AttributeError):
- print("SKIPPING (not connected)")
- 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:
- report(False, "GDB Exception: {}".format(sys.exc_info()[0]))
- print("All tests complete: %d failures" % n_failures)
- exit(n_failures)
-
-
-main()
+main(run_test)
--
2.43.0
^ permalink raw reply related [flat|nested] 11+ messages in thread
* [PATCH v3 3/3] tests/tcg: Add the PROT_NONE gdbstub test
2024-01-16 0:31 [PATCH v3 0/3] linux-user: Allow gdbstub to ignore page protection Ilya Leoshkevich
2024-01-16 0:31 ` [PATCH v3 1/3] " Ilya Leoshkevich
2024-01-16 0:31 ` [PATCH v3 2/3] tests/tcg: Factor out gdbstub test functions Ilya Leoshkevich
@ 2024-01-16 0:31 ` Ilya Leoshkevich
2024-01-22 15:43 ` Alex Bennée
2024-01-22 15:54 ` Alex Bennée
2 siblings, 2 replies; 11+ messages in thread
From: Ilya Leoshkevich @ 2024-01-16 0:31 UTC (permalink / raw)
To: Richard Henderson, Peter Maydell, Alex Bennée,
David Hildenbrand
Cc: Paolo Bonzini, Philippe Mathieu-Daudé, qemu-devel, qemu-arm,
qemu-s390x, Ilya Leoshkevich
Make sure that qemu gdbstub, like gdbserver, allows reading from and
writing to PROT_NONE pages.
Signed-off-by: Ilya Leoshkevich <iii@linux.ibm.com>
---
tests/tcg/multiarch/Makefile.target | 9 +++++-
tests/tcg/multiarch/gdbstub/prot-none.py | 22 +++++++++++++
tests/tcg/multiarch/prot-none.c | 40 ++++++++++++++++++++++++
3 files changed, 70 insertions(+), 1 deletion(-)
create mode 100644 tests/tcg/multiarch/gdbstub/prot-none.py
create mode 100644 tests/tcg/multiarch/prot-none.c
diff --git a/tests/tcg/multiarch/Makefile.target b/tests/tcg/multiarch/Makefile.target
index d31ba8d6ae4..315a2e13588 100644
--- a/tests/tcg/multiarch/Makefile.target
+++ b/tests/tcg/multiarch/Makefile.target
@@ -101,13 +101,20 @@ run-gdbstub-registers: sha512
--bin $< --test $(MULTIARCH_SRC)/gdbstub/registers.py, \
checking register enumeration)
+run-gdbstub-prot-none: prot-none
+ $(call run-test, $@, env PROT_NONE_PY=1 $(GDB_SCRIPT) \
+ --gdb $(GDB) \
+ --qemu $(QEMU) --qargs "$(QEMU_OPTS)" \
+ --bin $< --test $(MULTIARCH_SRC)/gdbstub/prot-none.py, \
+ accessing PROT_NONE memory)
+
else
run-gdbstub-%:
$(call skip-test, "gdbstub test $*", "need working gdb with $(patsubst -%,,$(TARGET_NAME)) support")
endif
EXTRA_RUNS += run-gdbstub-sha1 run-gdbstub-qxfer-auxv-read \
run-gdbstub-proc-mappings run-gdbstub-thread-breakpoint \
- run-gdbstub-registers
+ run-gdbstub-registers run-gdbstub-prot-none
# ARM Compatible Semi Hosting Tests
#
diff --git a/tests/tcg/multiarch/gdbstub/prot-none.py b/tests/tcg/multiarch/gdbstub/prot-none.py
new file mode 100644
index 00000000000..f1f1dd82cbe
--- /dev/null
+++ b/tests/tcg/multiarch/gdbstub/prot-none.py
@@ -0,0 +1,22 @@
+"""Test that GDB can access PROT_NONE pages.
+
+This runs as a sourced script (via -x, via run-test.py).
+
+SPDX-License-Identifier: GPL-2.0-or-later
+"""
+from test_gdbstub import main, report
+
+
+def run_test():
+ """Run through the tests one by one"""
+ gdb.Breakpoint("break_here")
+ gdb.execute("continue")
+ val = gdb.parse_and_eval("*(char[2] *)q").string()
+ report(val == "42", "{} == 42".format(val))
+ gdb.execute("set *(char[3] *)q = \"24\"")
+ gdb.execute("continue")
+ exitcode = int(gdb.parse_and_eval("$_exitcode"))
+ report(exitcode == 0, "{} == 0".format(exitcode))
+
+
+main(run_test)
diff --git a/tests/tcg/multiarch/prot-none.c b/tests/tcg/multiarch/prot-none.c
new file mode 100644
index 00000000000..dc56aadb3c5
--- /dev/null
+++ b/tests/tcg/multiarch/prot-none.c
@@ -0,0 +1,40 @@
+/*
+ * Test that GDB can access PROT_NONE pages.
+ *
+ * SPDX-License-Identifier: GPL-2.0-or-later
+ */
+#include <assert.h>
+#include <stdlib.h>
+#include <string.h>
+#include <sys/mman.h>
+#include <unistd.h>
+
+void break_here(void *q)
+{
+}
+
+int main(void)
+{
+ long pagesize = sysconf(_SC_PAGESIZE);
+ void *p, *q;
+ int err;
+
+ p = mmap(NULL, pagesize * 2, PROT_READ | PROT_WRITE,
+ MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
+ assert(p != MAP_FAILED);
+ q = p + pagesize - 1;
+ strcpy(q, "42");
+
+ err = mprotect(p, pagesize * 2, PROT_NONE);
+ assert(err == 0);
+
+ break_here(q);
+
+ err = mprotect(p, pagesize * 2, PROT_READ);
+ assert(err == 0);
+ if (getenv("PROT_NONE_PY")) {
+ assert(strcmp(q, "24") == 0);
+ }
+
+ return EXIT_SUCCESS;
+}
--
2.43.0
^ permalink raw reply related [flat|nested] 11+ messages in thread
* Re: [PATCH v3 3/3] tests/tcg: Add the PROT_NONE gdbstub test
2024-01-16 0:31 ` [PATCH v3 3/3] tests/tcg: Add the PROT_NONE gdbstub test Ilya Leoshkevich
@ 2024-01-22 15:43 ` Alex Bennée
2024-01-22 15:54 ` Alex Bennée
1 sibling, 0 replies; 11+ messages in thread
From: Alex Bennée @ 2024-01-22 15:43 UTC (permalink / raw)
To: Ilya Leoshkevich
Cc: Richard Henderson, Peter Maydell, David Hildenbrand,
Paolo Bonzini, Philippe Mathieu-Daudé, qemu-devel, qemu-arm,
qemu-s390x
Ilya Leoshkevich <iii@linux.ibm.com> writes:
> Make sure that qemu gdbstub, like gdbserver, allows reading from and
> writing to PROT_NONE pages.
>
> Signed-off-by: Ilya Leoshkevich <iii@linux.ibm.com>
Hmm I'm seeing the test hang and drop to the interactive python shell:
TEST basic gdbstub support on aarch64
Failed to read a valid object file image from memory.
qemu-aarch64: QEMU: Terminated via GDBstub
TEST basic gdbstub qXfer:auxv:read support on aarch64
Failed to read a valid object file image from memory.
qemu-aarch64: QEMU: Terminated via GDBstub
TEST proc mappings support on aarch64
Failed to read a valid object file image from memory.
qemu-aarch64: QEMU: Terminated via GDBstub
TEST hitting a breakpoint on non-main thread on aarch64
Failed to read a valid object file image from memory.
qemu-aarch64: QEMU: Terminated via GDBstub
TEST checking register enumeration on aarch64
Failed to read a valid object file image from memory.
qemu-aarch64: QEMU: Terminated via GDBstub
TEST accessing PROT_NONE memory on aarch64
Failed to read a valid object file image from memory.
Python 3.11.2 (main, Mar 13 2023, 12:18:29) [GCC 12.2.0] on linux
Type "help", "copyright", "credits" or "license" for more information.
(InteractiveConsole)
> ---
> tests/tcg/multiarch/Makefile.target | 9 +++++-
> tests/tcg/multiarch/gdbstub/prot-none.py | 22 +++++++++++++
> tests/tcg/multiarch/prot-none.c | 40 ++++++++++++++++++++++++
> 3 files changed, 70 insertions(+), 1 deletion(-)
> create mode 100644 tests/tcg/multiarch/gdbstub/prot-none.py
> create mode 100644 tests/tcg/multiarch/prot-none.c
>
> diff --git a/tests/tcg/multiarch/Makefile.target b/tests/tcg/multiarch/Makefile.target
> index d31ba8d6ae4..315a2e13588 100644
> --- a/tests/tcg/multiarch/Makefile.target
> +++ b/tests/tcg/multiarch/Makefile.target
> @@ -101,13 +101,20 @@ run-gdbstub-registers: sha512
> --bin $< --test $(MULTIARCH_SRC)/gdbstub/registers.py, \
> checking register enumeration)
>
> +run-gdbstub-prot-none: prot-none
> + $(call run-test, $@, env PROT_NONE_PY=1 $(GDB_SCRIPT) \
> + --gdb $(GDB) \
> + --qemu $(QEMU) --qargs "$(QEMU_OPTS)" \
> + --bin $< --test $(MULTIARCH_SRC)/gdbstub/prot-none.py, \
> + accessing PROT_NONE memory)
> +
> else
> run-gdbstub-%:
> $(call skip-test, "gdbstub test $*", "need working gdb with $(patsubst -%,,$(TARGET_NAME)) support")
> endif
> EXTRA_RUNS += run-gdbstub-sha1 run-gdbstub-qxfer-auxv-read \
> run-gdbstub-proc-mappings run-gdbstub-thread-breakpoint \
> - run-gdbstub-registers
> + run-gdbstub-registers run-gdbstub-prot-none
>
> # ARM Compatible Semi Hosting Tests
> #
> diff --git a/tests/tcg/multiarch/gdbstub/prot-none.py b/tests/tcg/multiarch/gdbstub/prot-none.py
> new file mode 100644
> index 00000000000..f1f1dd82cbe
> --- /dev/null
> +++ b/tests/tcg/multiarch/gdbstub/prot-none.py
> @@ -0,0 +1,22 @@
> +"""Test that GDB can access PROT_NONE pages.
> +
> +This runs as a sourced script (via -x, via run-test.py).
> +
> +SPDX-License-Identifier: GPL-2.0-or-later
> +"""
> +from test_gdbstub import main, report
> +
> +
> +def run_test():
> + """Run through the tests one by one"""
> + gdb.Breakpoint("break_here")
> + gdb.execute("continue")
> + val = gdb.parse_and_eval("*(char[2] *)q").string()
> + report(val == "42", "{} == 42".format(val))
> + gdb.execute("set *(char[3] *)q = \"24\"")
> + gdb.execute("continue")
> + exitcode = int(gdb.parse_and_eval("$_exitcode"))
> + report(exitcode == 0, "{} == 0".format(exitcode))
> +
> +
> +main(run_test)
> diff --git a/tests/tcg/multiarch/prot-none.c b/tests/tcg/multiarch/prot-none.c
> new file mode 100644
> index 00000000000..dc56aadb3c5
> --- /dev/null
> +++ b/tests/tcg/multiarch/prot-none.c
> @@ -0,0 +1,40 @@
> +/*
> + * Test that GDB can access PROT_NONE pages.
> + *
> + * SPDX-License-Identifier: GPL-2.0-or-later
> + */
> +#include <assert.h>
> +#include <stdlib.h>
> +#include <string.h>
> +#include <sys/mman.h>
> +#include <unistd.h>
> +
> +void break_here(void *q)
> +{
> +}
> +
> +int main(void)
> +{
> + long pagesize = sysconf(_SC_PAGESIZE);
> + void *p, *q;
> + int err;
> +
> + p = mmap(NULL, pagesize * 2, PROT_READ | PROT_WRITE,
> + MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
> + assert(p != MAP_FAILED);
> + q = p + pagesize - 1;
> + strcpy(q, "42");
> +
> + err = mprotect(p, pagesize * 2, PROT_NONE);
> + assert(err == 0);
> +
> + break_here(q);
> +
> + err = mprotect(p, pagesize * 2, PROT_READ);
> + assert(err == 0);
> + if (getenv("PROT_NONE_PY")) {
> + assert(strcmp(q, "24") == 0);
> + }
> +
> + return EXIT_SUCCESS;
> +}
--
Alex Bennée
Virtualisation Tech Lead @ Linaro
^ permalink raw reply [flat|nested] 11+ messages in thread
* Re: [PATCH v3 3/3] tests/tcg: Add the PROT_NONE gdbstub test
2024-01-16 0:31 ` [PATCH v3 3/3] tests/tcg: Add the PROT_NONE gdbstub test Ilya Leoshkevich
2024-01-22 15:43 ` Alex Bennée
@ 2024-01-22 15:54 ` Alex Bennée
2024-01-22 21:31 ` Ilya Leoshkevich
1 sibling, 1 reply; 11+ messages in thread
From: Alex Bennée @ 2024-01-22 15:54 UTC (permalink / raw)
To: Ilya Leoshkevich
Cc: Richard Henderson, Peter Maydell, David Hildenbrand,
Paolo Bonzini, Philippe Mathieu-Daudé, qemu-devel, qemu-arm,
qemu-s390x
Ilya Leoshkevich <iii@linux.ibm.com> writes:
> Make sure that qemu gdbstub, like gdbserver, allows reading from and
> writing to PROT_NONE pages.
>
> Signed-off-by: Ilya Leoshkevich <iii@linux.ibm.com>
> ---
> tests/tcg/multiarch/Makefile.target | 9 +++++-
> tests/tcg/multiarch/gdbstub/prot-none.py | 22 +++++++++++++
> tests/tcg/multiarch/prot-none.c | 40 ++++++++++++++++++++++++
> 3 files changed, 70 insertions(+), 1 deletion(-)
> create mode 100644 tests/tcg/multiarch/gdbstub/prot-none.py
> create mode 100644 tests/tcg/multiarch/prot-none.c
>
> diff --git a/tests/tcg/multiarch/Makefile.target b/tests/tcg/multiarch/Makefile.target
> index d31ba8d6ae4..315a2e13588 100644
> --- a/tests/tcg/multiarch/Makefile.target
> +++ b/tests/tcg/multiarch/Makefile.target
> @@ -101,13 +101,20 @@ run-gdbstub-registers: sha512
> --bin $< --test $(MULTIARCH_SRC)/gdbstub/registers.py, \
> checking register enumeration)
>
> +run-gdbstub-prot-none: prot-none
> + $(call run-test, $@, env PROT_NONE_PY=1 $(GDB_SCRIPT) \
> + --gdb $(GDB) \
> + --qemu $(QEMU) --qargs "$(QEMU_OPTS)" \
> + --bin $< --test $(MULTIARCH_SRC)/gdbstub/prot-none.py, \
> + accessing PROT_NONE memory)
> +
> else
> run-gdbstub-%:
> $(call skip-test, "gdbstub test $*", "need working gdb with $(patsubst -%,,$(TARGET_NAME)) support")
> endif
> EXTRA_RUNS += run-gdbstub-sha1 run-gdbstub-qxfer-auxv-read \
> run-gdbstub-proc-mappings run-gdbstub-thread-breakpoint \
> - run-gdbstub-registers
> + run-gdbstub-registers run-gdbstub-prot-none
>
> # ARM Compatible Semi Hosting Tests
> #
> diff --git a/tests/tcg/multiarch/gdbstub/prot-none.py b/tests/tcg/multiarch/gdbstub/prot-none.py
> new file mode 100644
> index 00000000000..f1f1dd82cbe
> --- /dev/null
> +++ b/tests/tcg/multiarch/gdbstub/prot-none.py
> @@ -0,0 +1,22 @@
> +"""Test that GDB can access PROT_NONE pages.
> +
> +This runs as a sourced script (via -x, via run-test.py).
> +
> +SPDX-License-Identifier: GPL-2.0-or-later
> +"""
> +from test_gdbstub import main, report
> +
> +
> +def run_test():
> + """Run through the tests one by one"""
> + gdb.Breakpoint("break_here")
> + gdb.execute("continue")
> + val = gdb.parse_and_eval("*(char[2] *)q").string()
Better traceback:
Breakpoint 1, break_here (q=0x400000802fff) at /home/alex/lsrc/qemu.git/tests/tcg/multiarch/prot-none.c:14
14 }
GDB Exception:
Traceback (most recent call last):
File "/home/alex/lsrc/qemu.git/tests/guest-debug/test_gdbstub.py", line 42, in main
test()
File "./tests/tcg/multiarch/gdbstub/prot-none.py", line 14, in run_test
val = gdb.parse_and_eval("*(char[2] *)q").string()
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
gdb.MemoryError: Cannot access memory at address 0x400000802fff
Python 3.11.2 (main, Mar 13 2023, 12:18:29) [GCC 12.2.0] on linux
Type "help", "copyright", "credits" or "license" for more information.
(InteractiveConsole)
>>>
> + report(val == "42", "{} == 42".format(val))
> + gdb.execute("set *(char[3] *)q = \"24\"")
> + gdb.execute("continue")
> + exitcode = int(gdb.parse_and_eval("$_exitcode"))
> + report(exitcode == 0, "{} == 0".format(exitcode))
> +
> +
> +main(run_test)
> diff --git a/tests/tcg/multiarch/prot-none.c b/tests/tcg/multiarch/prot-none.c
> new file mode 100644
> index 00000000000..dc56aadb3c5
> --- /dev/null
> +++ b/tests/tcg/multiarch/prot-none.c
> @@ -0,0 +1,40 @@
> +/*
> + * Test that GDB can access PROT_NONE pages.
> + *
> + * SPDX-License-Identifier: GPL-2.0-or-later
> + */
> +#include <assert.h>
> +#include <stdlib.h>
> +#include <string.h>
> +#include <sys/mman.h>
> +#include <unistd.h>
> +
> +void break_here(void *q)
> +{
> +}
> +
> +int main(void)
> +{
> + long pagesize = sysconf(_SC_PAGESIZE);
> + void *p, *q;
> + int err;
> +
> + p = mmap(NULL, pagesize * 2, PROT_READ | PROT_WRITE,
> + MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
> + assert(p != MAP_FAILED);
> + q = p + pagesize - 1;
> + strcpy(q, "42");
> +
> + err = mprotect(p, pagesize * 2, PROT_NONE);
> + assert(err == 0);
> +
> + break_here(q);
> +
> + err = mprotect(p, pagesize * 2, PROT_READ);
> + assert(err == 0);
> + if (getenv("PROT_NONE_PY")) {
> + assert(strcmp(q, "24") == 0);
> + }
> +
> + return EXIT_SUCCESS;
> +}
--
Alex Bennée
Virtualisation Tech Lead @ Linaro
^ permalink raw reply [flat|nested] 11+ messages in thread
* Re: [PATCH v3 2/3] tests/tcg: Factor out gdbstub test functions
2024-01-16 0:31 ` [PATCH v3 2/3] tests/tcg: Factor out gdbstub test functions Ilya Leoshkevich
@ 2024-01-22 16:00 ` Alex Bennée
2024-01-22 21:08 ` Ilya Leoshkevich
0 siblings, 1 reply; 11+ messages in thread
From: Alex Bennée @ 2024-01-22 16:00 UTC (permalink / raw)
To: Ilya Leoshkevich
Cc: Richard Henderson, Peter Maydell, David Hildenbrand,
Paolo Bonzini, Philippe Mathieu-Daudé, qemu-devel, qemu-arm,
qemu-s390x
Ilya Leoshkevich <iii@linux.ibm.com> writes:
> Both the report() function as well as the initial gdbstub test sequence
> are copy-pasted into ~10 files with slight modifications. This
> indicates that they are indeed generic, so factor them out. While
> at it, add a few newlines to make the formatting closer to PEP-8.
>
> Signed-off-by: Ilya Leoshkevich <iii@linux.ibm.com>
> ---
> tests/guest-debug/run-test.py | 7 ++-
> tests/guest-debug/test_gdbstub.py | 58 +++++++++++++++++++
> tests/tcg/aarch64/gdbstub/test-sve-ioctl.py | 34 +----------
> tests/tcg/aarch64/gdbstub/test-sve.py | 33 +----------
> tests/tcg/multiarch/gdbstub/interrupt.py | 47 ++-------------
> tests/tcg/multiarch/gdbstub/memory.py | 41 +------------
> tests/tcg/multiarch/gdbstub/registers.py | 41 ++-----------
> tests/tcg/multiarch/gdbstub/sha1.py | 40 ++-----------
> .../multiarch/gdbstub/test-proc-mappings.py | 39 +------------
> .../multiarch/gdbstub/test-qxfer-auxv-read.py | 37 +-----------
> .../gdbstub/test-thread-breakpoint.py | 37 +-----------
> tests/tcg/s390x/gdbstub/test-signals-s390x.py | 42 +-------------
> tests/tcg/s390x/gdbstub/test-svc.py | 39 +------------
> 13 files changed, 98 insertions(+), 397 deletions(-)
> create mode 100644 tests/guest-debug/test_gdbstub.py
>
> diff --git a/tests/guest-debug/run-test.py b/tests/guest-debug/run-test.py
> index b13b27d4b19..368ff8a8903 100755
> --- a/tests/guest-debug/run-test.py
> +++ b/tests/guest-debug/run-test.py
> @@ -97,7 +97,12 @@ def log(output, msg):
> sleep(1)
> log(output, "GDB CMD: %s" % (gdb_cmd))
>
> - result = subprocess.call(gdb_cmd, shell=True, stdout=output, stderr=stderr)
> + 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)
>
> # 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
> diff --git a/tests/guest-debug/test_gdbstub.py b/tests/guest-debug/test_gdbstub.py
> new file mode 100644
> index 00000000000..a71cdaa915a
> --- /dev/null
> +++ b/tests/guest-debug/test_gdbstub.py
> @@ -0,0 +1,58 @@
> +"""Helper functions for gdbstub testing
> +
> +"""
> +from __future__ import print_function
> +import gdb
> +import sys
> +import traceback
> +
> +fail_count = 0
> +
> +
> +def report(cond, msg):
> + """Report success/fail of a test"""
> + if cond:
> + print("PASS: {}".format(msg))
> + else:
> + print("FAIL: {}".format(msg))
> + global fail_count
> + fail_count += 1
> +
> +
> +def main(test, expected_arch=None):
> + """Run a test function
> +
> + This runs as the script it sourced (via -x, via run-test.py)."""
> + try:
> + inferior = gdb.selected_inferior()
> + arch = inferior.architecture()
> + print("ATTACHED: {}".format(arch.name()))
> + if expected_arch is not None:
> + report(arch.name() == expected_arch,
> + "connected to {}".format(expected_arch))
> + except (gdb.error, AttributeError):
> + print("SKIP: not connected")
> + exit(0)
> +
> + if gdb.parse_and_eval("$pc") == 0:
> + print("SKIP: PC not set")
> + exit(0)
> +
> + try:
> + test()
> + except:
> + print("GDB Exception:")
> + traceback.print_exc(file=sys.stdout)
> + global fail_count
> + fail_count += 1
> + import code
> + code.InteractiveConsole(locals=globals()).interact()
> + raise
While I can see this is useful we don't want to default to an
interactive console as that will hang the test in CI type setups. Can we
make this a option we enable?
> +
> + try:
> + gdb.execute("kill")
> + except gdb.error:
> + pass
> +
> + print("All tests complete: {} failures".format(fail_count))
> + exit(fail_count)
> diff --git a/tests/tcg/aarch64/gdbstub/test-sve-ioctl.py b/tests/tcg/aarch64/gdbstub/test-sve-ioctl.py
> index ee8d467e59d..a78a3a2514d 100644
> --- a/tests/tcg/aarch64/gdbstub/test-sve-ioctl.py
> +++ b/tests/tcg/aarch64/gdbstub/test-sve-ioctl.py
> @@ -8,19 +8,10 @@
> #
>
> import gdb
> -import sys
> +from test_gdbstub import main, report
>
> initial_vlen = 0
> -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
>
> class TestBreakpoint(gdb.Breakpoint):
> def __init__(self, sym_name="__sve_ld_done"):
> @@ -64,26 +55,5 @@ def run_test():
>
> gdb.execute("c")
>
> -#
> -# This runs as the script it sourced (via -x, via run-test.py)
> -#
> -try:
> - inferior = gdb.selected_inferior()
> - arch = inferior.architecture()
> - report(arch.name() == "aarch64", "connected to aarch64")
> -except (gdb.error, AttributeError):
> - print("SKIPPING (not connected)", file=sys.stderr)
> - exit(0)
> -
> -try:
> - # Run the actual tests
> - run_test()
> -except:
> - print ("GDB Exception: %s" % (sys.exc_info()[0]))
> - failcount += 1
> - import code
> - code.InteractiveConsole(locals=globals()).interact()
> - raise
>
> -print("All tests complete: %d failures" % failcount)
> -exit(failcount)
> +main(run_test, expected_arch="aarch64")
> diff --git a/tests/tcg/aarch64/gdbstub/test-sve.py b/tests/tcg/aarch64/gdbstub/test-sve.py
> index afd8ece98dd..84cdcd4a32e 100644
> --- a/tests/tcg/aarch64/gdbstub/test-sve.py
> +++ b/tests/tcg/aarch64/gdbstub/test-sve.py
> @@ -6,20 +6,10 @@
> #
>
> import gdb
> -import sys
> +from test_gdbstub import main, report
>
> MAGIC = 0xDEADBEEF
>
> -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 run_test():
> "Run through the tests one by one"
> @@ -54,24 +44,5 @@ def run_test():
> report(str(v.type) == "uint64_t", "size of %s" % (reg))
> report(int(v) == MAGIC, "%s is 0x%x" % (reg, MAGIC))
>
> -#
> -# This runs as the script it sourced (via -x, via run-test.py)
> -#
> -try:
> - inferior = gdb.selected_inferior()
> - arch = inferior.architecture()
> - report(arch.name() == "aarch64", "connected to aarch64")
> -except (gdb.error, AttributeError):
> - print("SKIPPING (not connected)", file=sys.stderr)
> - exit(0)
> -
> -try:
> - # Run the actual tests
> - run_test()
> -except:
> - print ("GDB Exception: %s" % (sys.exc_info()[0]))
> - failcount += 1
> -
> -print("All tests complete: %d failures" % failcount)
>
> -exit(failcount)
> +main(run_test, expected_arch="aarch64")
> diff --git a/tests/tcg/multiarch/gdbstub/interrupt.py b/tests/tcg/multiarch/gdbstub/interrupt.py
> index c016e7afbbf..90a45b5140a 100644
> --- a/tests/tcg/multiarch/gdbstub/interrupt.py
> +++ b/tests/tcg/multiarch/gdbstub/interrupt.py
> @@ -8,19 +8,7 @@
> #
>
> 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
> +from test_gdbstub import main, report
>
>
> def check_interrupt(thread):
> @@ -59,6 +47,9 @@ def run_test():
> Test if interrupting the code always lands us on the same thread when
> running with scheduler-lock enabled.
> """
> + if len(gdb.selected_inferior().threads()) == 1:
> + print("SKIP: set to run on a single thread")
> + exit(0)
>
> gdb.execute("set scheduler-locking on")
> for thread in gdb.selected_inferior().threads():
> @@ -66,32 +57,4 @@ def run_test():
> "thread %d resumes correctly on interrupt" % thread.num)
>
>
> -#
> -# 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)
> -if len(gdb.selected_inferior().threads()) == 1:
> - print("SKIP: set to run on a single thread")
> - 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)
> +main(run_test)
> diff --git a/tests/tcg/multiarch/gdbstub/memory.py b/tests/tcg/multiarch/gdbstub/memory.py
> index fb1d06b7bb7..532b92e7fb3 100644
> --- a/tests/tcg/multiarch/gdbstub/memory.py
> +++ b/tests/tcg/multiarch/gdbstub/memory.py
> @@ -9,18 +9,7 @@
>
> 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
> +from test_gdbstub import main, report
>
>
> def check_step():
> @@ -99,29 +88,5 @@ def run_test():
>
> 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)
> +
> +main(run_test)
> diff --git a/tests/tcg/multiarch/gdbstub/registers.py b/tests/tcg/multiarch/gdbstub/registers.py
> index 688c0611072..b3d13cb077f 100644
> --- a/tests/tcg/multiarch/gdbstub/registers.py
> +++ b/tests/tcg/multiarch/gdbstub/registers.py
> @@ -7,20 +7,11 @@
> # SPDX-License-Identifier: GPL-2.0-or-later
>
> import gdb
> -import sys
> import xml.etree.ElementTree as ET
> +from test_gdbstub import main, report
>
> -initial_vlen = 0
> -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
> +initial_vlen = 0
>
>
> def fetch_xml_regmap():
> @@ -75,6 +66,7 @@ def fetch_xml_regmap():
>
> return reg_map
>
> +
> def get_register_by_regnum(reg_map, regnum):
> """
> Helper to find a register from the map via its XML regnum
> @@ -84,6 +76,7 @@ def get_register_by_regnum(reg_map, regnum):
> return entry
> return None
>
> +
> def crosscheck_remote_xml(reg_map):
> """
> Cross-check the list of remote-registers with the XML info.
> @@ -144,6 +137,7 @@ def crosscheck_remote_xml(reg_map):
> elif "seen" not in x_reg:
> print(f"{x_reg} wasn't seen in remote-registers")
>
> +
> def initial_register_read(reg_map):
> """
> Do an initial read of all registers that we know gdb cares about
> @@ -214,27 +208,4 @@ def run_test():
> complete_and_diff(reg_map)
>
>
> -#
> -# 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_test()
> -except (gdb.error):
> - print ("GDB Exception: %s" % (sys.exc_info()[0]))
> - failcount += 1
> - pass
> -
> -print("All tests complete: %d failures" % failcount)
> -exit(failcount)
> +main(run_test)
> diff --git a/tests/tcg/multiarch/gdbstub/sha1.py b/tests/tcg/multiarch/gdbstub/sha1.py
> index 416728415f9..1ce711a402c 100644
> --- a/tests/tcg/multiarch/gdbstub/sha1.py
> +++ b/tests/tcg/multiarch/gdbstub/sha1.py
> @@ -7,19 +7,11 @@
> #
>
> import gdb
> -import sys
> +from test_gdbstub import main, report
> +
>
> initial_vlen = 0
> -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_break(sym_name):
> "Setup breakpoint, continue and check we stopped."
> @@ -35,6 +27,7 @@ def check_break(sym_name):
>
> bp.delete()
>
> +
> def run_test():
> "Run through the tests one by one"
>
> @@ -57,28 +50,5 @@ def run_test():
> # finally check we don't barf inspecting registers
> gdb.execute("info registers")
>
> -#
> -# 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
> -
> -print("All tests complete: %d failures" % failcount)
> -exit(failcount)
> +
> +main(run_test)
> diff --git a/tests/tcg/multiarch/gdbstub/test-proc-mappings.py b/tests/tcg/multiarch/gdbstub/test-proc-mappings.py
> index 04ec61d2197..564613fabf0 100644
> --- a/tests/tcg/multiarch/gdbstub/test-proc-mappings.py
> +++ b/tests/tcg/multiarch/gdbstub/test-proc-mappings.py
> @@ -3,20 +3,7 @@
> This runs as a sourced script (via -x, via run-test.py)."""
> from __future__ import print_function
> import gdb
> -import sys
> -
> -
> -n_failures = 0
> -
> -
> -def report(cond, msg):
> - """Report success/fail of a test"""
> - if cond:
> - print("PASS: {}".format(msg))
> - else:
> - print("FAIL: {}".format(msg))
> - global n_failures
> - n_failures += 1
> +from test_gdbstub import main, report
>
>
> def run_test():
> @@ -37,26 +24,4 @@ def run_test():
> # report("/sha1" in mappings, "Found the test binary name in the mappings")
>
>
> -def main():
> - """Prepare the environment and run through the tests"""
> - try:
> - inferior = gdb.selected_inferior()
> - print("ATTACHED: {}".format(inferior.architecture().name()))
> - except (gdb.error, AttributeError):
> - print("SKIPPING (not connected)")
> - 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:
> - report(False, "GDB Exception: {}".format(sys.exc_info()[0]))
> - print("All tests complete: %d failures" % n_failures)
> - exit(n_failures)
> -
> -
> -main()
> +main(run_test)
> diff --git a/tests/tcg/multiarch/gdbstub/test-qxfer-auxv-read.py b/tests/tcg/multiarch/gdbstub/test-qxfer-auxv-read.py
> index 926fa962b77..00c26ab4a95 100644
> --- a/tests/tcg/multiarch/gdbstub/test-qxfer-auxv-read.py
> +++ b/tests/tcg/multiarch/gdbstub/test-qxfer-auxv-read.py
> @@ -6,18 +6,8 @@
> #
>
> import gdb
> -import sys
> +from test_gdbstub import main, report
>
> -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 run_test():
> "Run through the tests one by one"
> @@ -26,28 +16,5 @@ def run_test():
> report(isinstance(auxv, str), "Fetched auxv from inferior")
> report(auxv.find("sha1"), "Found test binary name in auxv")
>
> -#
> -# 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
>
> -print("All tests complete: %d failures" % failcount)
> -exit(failcount)
> +main(run_test)
> diff --git a/tests/tcg/multiarch/gdbstub/test-thread-breakpoint.py b/tests/tcg/multiarch/gdbstub/test-thread-breakpoint.py
> index e57d2a8db8b..4d6b6b9fbe7 100644
> --- a/tests/tcg/multiarch/gdbstub/test-thread-breakpoint.py
> +++ b/tests/tcg/multiarch/gdbstub/test-thread-breakpoint.py
> @@ -6,18 +6,8 @@
> #
>
> import gdb
> -import sys
> +from test_gdbstub import main, report
>
> -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 run_test():
> "Run through the tests one by one"
> @@ -29,28 +19,5 @@ def run_test():
> frame = gdb.selected_frame()
> report(str(frame.function()) == "thread1_func", "break @ %s"%frame)
>
> -#
> -# 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
>
> -print("All tests complete: %d failures" % failcount)
> -exit(failcount)
> +main(run_test)
> diff --git a/tests/tcg/s390x/gdbstub/test-signals-s390x.py b/tests/tcg/s390x/gdbstub/test-signals-s390x.py
> index ca2bbc0b03e..b6b7b39fc46 100644
> --- a/tests/tcg/s390x/gdbstub/test-signals-s390x.py
> +++ b/tests/tcg/s390x/gdbstub/test-signals-s390x.py
> @@ -7,19 +7,7 @@
> #
>
> 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
> +from test_gdbstub import main, report
>
>
> def run_test():
> @@ -42,31 +30,7 @@ def run_test():
> gdb.Breakpoint("_exit")
> gdb.execute("c")
> status = int(gdb.parse_and_eval("$r2"))
> - report(status == 0, "status == 0");
> -
> -
> -#
> -# 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)
> + report(status == 0, "status == 0")
>
> -try:
> - # Run the actual tests
> - run_test()
> -except (gdb.error):
> - print("GDB Exception: %s" % (sys.exc_info()[0]))
> - failcount += 1
> - pass
>
> -print("All tests complete: %d failures" % failcount)
> -exit(failcount)
> +main(run_test)
> diff --git a/tests/tcg/s390x/gdbstub/test-svc.py b/tests/tcg/s390x/gdbstub/test-svc.py
> index 804705fede9..17210b4e020 100644
> --- a/tests/tcg/s390x/gdbstub/test-svc.py
> +++ b/tests/tcg/s390x/gdbstub/test-svc.py
> @@ -3,20 +3,7 @@
> This runs as a sourced script (via -x, via run-test.py)."""
> from __future__ import print_function
> import gdb
> -import sys
> -
> -
> -n_failures = 0
> -
> -
> -def report(cond, msg):
> - """Report success/fail of a test"""
> - if cond:
> - print("PASS: {}".format(msg))
> - else:
> - print("FAIL: {}".format(msg))
> - global n_failures
> - n_failures += 1
> +from test_gdbstub import main, report
>
>
> def run_test():
> @@ -35,26 +22,4 @@ def run_test():
> gdb.execute("si")
>
>
> -def main():
> - """Prepare the environment and run through the tests"""
> - try:
> - inferior = gdb.selected_inferior()
> - print("ATTACHED: {}".format(inferior.architecture().name()))
> - except (gdb.error, AttributeError):
> - print("SKIPPING (not connected)")
> - 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:
> - report(False, "GDB Exception: {}".format(sys.exc_info()[0]))
> - print("All tests complete: %d failures" % n_failures)
> - exit(n_failures)
> -
> -
> -main()
> +main(run_test)
--
Alex Bennée
Virtualisation Tech Lead @ Linaro
^ permalink raw reply [flat|nested] 11+ messages in thread
* Re: Re: [PATCH v3 2/3] tests/tcg: Factor out gdbstub test functions
2024-01-22 16:00 ` Alex Bennée
@ 2024-01-22 21:08 ` Ilya Leoshkevich
0 siblings, 0 replies; 11+ messages in thread
From: Ilya Leoshkevich @ 2024-01-22 21:08 UTC (permalink / raw)
To: Alex Bennée
Cc: Richard Henderson, Peter Maydell, David Hildenbrand,
Paolo Bonzini, Philippe Mathieu-Daudé, qemu-devel, qemu-arm,
qemu-s390x
On Mon, Jan 22, 2024 at 04:00:44PM +0000, Alex Bennée wrote:
> Ilya Leoshkevich <iii@linux.ibm.com> writes:
>
> > Both the report() function as well as the initial gdbstub test sequence
> > are copy-pasted into ~10 files with slight modifications. This
> > indicates that they are indeed generic, so factor them out. While
> > at it, add a few newlines to make the formatting closer to PEP-8.
> >
> > Signed-off-by: Ilya Leoshkevich <iii@linux.ibm.com>
> > ---
> > tests/guest-debug/run-test.py | 7 ++-
> > tests/guest-debug/test_gdbstub.py | 58 +++++++++++++++++++
> > tests/tcg/aarch64/gdbstub/test-sve-ioctl.py | 34 +----------
> > tests/tcg/aarch64/gdbstub/test-sve.py | 33 +----------
> > tests/tcg/multiarch/gdbstub/interrupt.py | 47 ++-------------
> > tests/tcg/multiarch/gdbstub/memory.py | 41 +------------
> > tests/tcg/multiarch/gdbstub/registers.py | 41 ++-----------
> > tests/tcg/multiarch/gdbstub/sha1.py | 40 ++-----------
> > .../multiarch/gdbstub/test-proc-mappings.py | 39 +------------
> > .../multiarch/gdbstub/test-qxfer-auxv-read.py | 37 +-----------
> > .../gdbstub/test-thread-breakpoint.py | 37 +-----------
> > tests/tcg/s390x/gdbstub/test-signals-s390x.py | 42 +-------------
> > tests/tcg/s390x/gdbstub/test-svc.py | 39 +------------
> > 13 files changed, 98 insertions(+), 397 deletions(-)
> > create mode 100644 tests/guest-debug/test_gdbstub.py
[...]
> > + if gdb.parse_and_eval("$pc") == 0:
> > + print("SKIP: PC not set")
> > + exit(0)
> > +
> > + try:
> > + test()
> > + except:
> > + print("GDB Exception:")
> > + traceback.print_exc(file=sys.stdout)
> > + global fail_count
> > + fail_count += 1
> > + import code
> > + code.InteractiveConsole(locals=globals()).interact()
> > + raise
>
> While I can see this is useful we don't want to default to an
> interactive console as that will hang the test in CI type setups. Can we
> make this a option we enable?
Would something like `export QEMU_TEST_INTERACTIVE=1` be okay?
[...]
^ permalink raw reply [flat|nested] 11+ messages in thread
* Re: Re: [PATCH v3 3/3] tests/tcg: Add the PROT_NONE gdbstub test
2024-01-22 15:54 ` Alex Bennée
@ 2024-01-22 21:31 ` Ilya Leoshkevich
2024-01-22 23:19 ` Alex Bennée
0 siblings, 1 reply; 11+ messages in thread
From: Ilya Leoshkevich @ 2024-01-22 21:31 UTC (permalink / raw)
To: Alex Bennée
Cc: Richard Henderson, Peter Maydell, David Hildenbrand,
Paolo Bonzini, Philippe Mathieu-Daudé, qemu-devel, qemu-arm,
qemu-s390x
On Mon, Jan 22, 2024 at 03:54:32PM +0000, Alex Bennée wrote:
> Ilya Leoshkevich <iii@linux.ibm.com> writes:
>
> > Make sure that qemu gdbstub, like gdbserver, allows reading from and
> > writing to PROT_NONE pages.
> >
> > Signed-off-by: Ilya Leoshkevich <iii@linux.ibm.com>
> > ---
> > tests/tcg/multiarch/Makefile.target | 9 +++++-
> > tests/tcg/multiarch/gdbstub/prot-none.py | 22 +++++++++++++
> > tests/tcg/multiarch/prot-none.c | 40 ++++++++++++++++++++++++
> > 3 files changed, 70 insertions(+), 1 deletion(-)
> > create mode 100644 tests/tcg/multiarch/gdbstub/prot-none.py
> > create mode 100644 tests/tcg/multiarch/prot-none.c
[...]
> > +def run_test():
> > + """Run through the tests one by one"""
> > + gdb.Breakpoint("break_here")
> > + gdb.execute("continue")
> > + val = gdb.parse_and_eval("*(char[2] *)q").string()
>
> Better traceback:
>
> Breakpoint 1, break_here (q=0x400000802fff) at /home/alex/lsrc/qemu.git/tests/tcg/multiarch/prot-none.c:14
> 14 }
> GDB Exception:
> Traceback (most recent call last):
> File "/home/alex/lsrc/qemu.git/tests/guest-debug/test_gdbstub.py", line 42, in main
> test()
> File "./tests/tcg/multiarch/gdbstub/prot-none.py", line 14, in run_test
> val = gdb.parse_and_eval("*(char[2] *)q").string()
> ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
> gdb.MemoryError: Cannot access memory at address 0x400000802fff
> Python 3.11.2 (main, Mar 13 2023, 12:18:29) [GCC 12.2.0] on linux
> Type "help", "copyright", "credits" or "license" for more information.
> (InteractiveConsole)
> >>>
Thanks for the debug output. This shows that the feature being tested
doesn't work (the value of `q` looks sane to me). May I ask what host
distro is this? I tried on x86_64 Fedora 38 and x86_64 Ubuntu 22.04 so
far, and the test was successful.
[...]
^ permalink raw reply [flat|nested] 11+ messages in thread
* Re: [PATCH v3 3/3] tests/tcg: Add the PROT_NONE gdbstub test
2024-01-22 21:31 ` Ilya Leoshkevich
@ 2024-01-22 23:19 ` Alex Bennée
2024-01-25 2:39 ` Ilya Leoshkevich
0 siblings, 1 reply; 11+ messages in thread
From: Alex Bennée @ 2024-01-22 23:19 UTC (permalink / raw)
To: Ilya Leoshkevich
Cc: Richard Henderson, Peter Maydell, David Hildenbrand,
Paolo Bonzini, Philippe Mathieu-Daudé, qemu-devel, qemu-arm,
qemu-s390x
Ilya Leoshkevich <iii@linux.ibm.com> writes:
> On Mon, Jan 22, 2024 at 03:54:32PM +0000, Alex Bennée wrote:
>> Ilya Leoshkevich <iii@linux.ibm.com> writes:
>>
>> > Make sure that qemu gdbstub, like gdbserver, allows reading from and
>> > writing to PROT_NONE pages.
>> >
>> > Signed-off-by: Ilya Leoshkevich <iii@linux.ibm.com>
>> > ---
>> > tests/tcg/multiarch/Makefile.target | 9 +++++-
>> > tests/tcg/multiarch/gdbstub/prot-none.py | 22 +++++++++++++
>> > tests/tcg/multiarch/prot-none.c | 40 ++++++++++++++++++++++++
>> > 3 files changed, 70 insertions(+), 1 deletion(-)
>> > create mode 100644 tests/tcg/multiarch/gdbstub/prot-none.py
>> > create mode 100644 tests/tcg/multiarch/prot-none.c
>
> [...]
>
>> > +def run_test():
>> > + """Run through the tests one by one"""
>> > + gdb.Breakpoint("break_here")
>> > + gdb.execute("continue")
>> > + val = gdb.parse_and_eval("*(char[2] *)q").string()
>>
>> Better traceback:
>>
>> Breakpoint 1, break_here (q=0x400000802fff) at /home/alex/lsrc/qemu.git/tests/tcg/multiarch/prot-none.c:14
>> 14 }
>> GDB Exception:
>> Traceback (most recent call last):
>> File "/home/alex/lsrc/qemu.git/tests/guest-debug/test_gdbstub.py", line 42, in main
>> test()
>> File "./tests/tcg/multiarch/gdbstub/prot-none.py", line 14, in run_test
>> val = gdb.parse_and_eval("*(char[2] *)q").string()
>> ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
>> gdb.MemoryError: Cannot access memory at address 0x400000802fff
>> Python 3.11.2 (main, Mar 13 2023, 12:18:29) [GCC 12.2.0] on linux
>> Type "help", "copyright", "credits" or "license" for more information.
>> (InteractiveConsole)
>> >>>
>
> Thanks for the debug output. This shows that the feature being tested
> doesn't work (the value of `q` looks sane to me). May I ask what host
> distro is this? I tried on x86_64 Fedora 38 and x86_64 Ubuntu 22.04 so
> far, and the test was successful.
Debian Bookworm (x86_64) with gdb-multiarch installed.
>
> [...]
--
Alex Bennée
Virtualisation Tech Lead @ Linaro
^ permalink raw reply [flat|nested] 11+ messages in thread
* Re: [PATCH v3 3/3] tests/tcg: Add the PROT_NONE gdbstub test
2024-01-22 23:19 ` Alex Bennée
@ 2024-01-25 2:39 ` Ilya Leoshkevich
0 siblings, 0 replies; 11+ messages in thread
From: Ilya Leoshkevich @ 2024-01-25 2:39 UTC (permalink / raw)
To: Alex Bennée
Cc: Richard Henderson, Peter Maydell, David Hildenbrand,
Paolo Bonzini, Philippe Mathieu-Daudé, qemu-devel, qemu-arm,
qemu-s390x
On Mon, Jan 22, 2024 at 11:19:05PM +0000, Alex Bennée wrote:
> Ilya Leoshkevich <iii@linux.ibm.com> writes:
>
> > On Mon, Jan 22, 2024 at 03:54:32PM +0000, Alex Bennée wrote:
> >> Ilya Leoshkevich <iii@linux.ibm.com> writes:
> >>
> >> > Make sure that qemu gdbstub, like gdbserver, allows reading from and
> >> > writing to PROT_NONE pages.
> >> >
> >> > Signed-off-by: Ilya Leoshkevich <iii@linux.ibm.com>
> >> > ---
> >> > tests/tcg/multiarch/Makefile.target | 9 +++++-
> >> > tests/tcg/multiarch/gdbstub/prot-none.py | 22 +++++++++++++
> >> > tests/tcg/multiarch/prot-none.c | 40 ++++++++++++++++++++++++
> >> > 3 files changed, 70 insertions(+), 1 deletion(-)
> >> > create mode 100644 tests/tcg/multiarch/gdbstub/prot-none.py
> >> > create mode 100644 tests/tcg/multiarch/prot-none.c
> >
> > [...]
> >
> >> > +def run_test():
> >> > + """Run through the tests one by one"""
> >> > + gdb.Breakpoint("break_here")
> >> > + gdb.execute("continue")
> >> > + val = gdb.parse_and_eval("*(char[2] *)q").string()
> >>
> >> Better traceback:
> >>
> >> Breakpoint 1, break_here (q=0x400000802fff) at /home/alex/lsrc/qemu.git/tests/tcg/multiarch/prot-none.c:14
> >> 14 }
> >> GDB Exception:
> >> Traceback (most recent call last):
> >> File "/home/alex/lsrc/qemu.git/tests/guest-debug/test_gdbstub.py", line 42, in main
> >> test()
> >> File "./tests/tcg/multiarch/gdbstub/prot-none.py", line 14, in run_test
> >> val = gdb.parse_and_eval("*(char[2] *)q").string()
> >> ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
> >> gdb.MemoryError: Cannot access memory at address 0x400000802fff
> >> Python 3.11.2 (main, Mar 13 2023, 12:18:29) [GCC 12.2.0] on linux
> >> Type "help", "copyright", "credits" or "license" for more information.
> >> (InteractiveConsole)
> >> >>>
> >
> > Thanks for the debug output. This shows that the feature being tested
> > doesn't work (the value of `q` looks sane to me). May I ask what host
> > distro is this? I tried on x86_64 Fedora 38 and x86_64 Ubuntu 22.04 so
> > far, and the test was successful.
>
> Debian Bookworm (x86_64) with gdb-multiarch installed.
> >
> > [...]
>
> --
> Alex Bennée
> Virtualisation Tech Lead @ Linaro
Hm, I tried that (in a VM, in case the kernel is somehow involved) and
that worked too:
Breakpoint 1, break_here (q=0x400000802fff) at /qemu/tests/tcg/multiarch/prot-none.c:14
14 }
PASS: 42 == 42
I wonder what else can be different. Can it be that in your case the
test runs without /proc? Or perhaps some additional LSM is enabled?
^ permalink raw reply [flat|nested] 11+ messages in thread
end of thread, other threads:[~2024-01-25 2:41 UTC | newest]
Thread overview: 11+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2024-01-16 0:31 [PATCH v3 0/3] linux-user: Allow gdbstub to ignore page protection Ilya Leoshkevich
2024-01-16 0:31 ` [PATCH v3 1/3] " Ilya Leoshkevich
2024-01-16 0:31 ` [PATCH v3 2/3] tests/tcg: Factor out gdbstub test functions Ilya Leoshkevich
2024-01-22 16:00 ` Alex Bennée
2024-01-22 21:08 ` Ilya Leoshkevich
2024-01-16 0:31 ` [PATCH v3 3/3] tests/tcg: Add the PROT_NONE gdbstub test Ilya Leoshkevich
2024-01-22 15:43 ` Alex Bennée
2024-01-22 15:54 ` Alex Bennée
2024-01-22 21:31 ` Ilya Leoshkevich
2024-01-22 23:19 ` Alex Bennée
2024-01-25 2:39 ` Ilya Leoshkevich
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).