From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: Received: from mail-wm1-f52.google.com (mail-wm1-f52.google.com [209.85.128.52]) by sourceware.org (Postfix) with ESMTPS id 6C426385E83E for ; Wed, 1 Dec 2021 14:32:12 +0000 (GMT) DMARC-Filter: OpenDMARC Filter v1.4.1 sourceware.org 6C426385E83E Received: by mail-wm1-f52.google.com with SMTP id c6-20020a05600c0ac600b0033c3aedd30aso1291606wmr.5 for ; Wed, 01 Dec 2021 06:32:12 -0800 (PST) X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20210112; h=x-gm-message-state:from:to:cc:subject:date:message-id:in-reply-to :references:mime-version:content-transfer-encoding; bh=dXYjSvHdQs8Ksip6SdJsay6JHGkcQ6JdUdAj7YroO0A=; b=HHDJlP+giuen7gRY4GJmWRq/1GJ5ZeEwANzCg64tuxVh7ZL7N9av5bB0oE896tL/ii //HEO3+BVAHSbGw83Y6AM1t2WuCwlEYiG8ZIvJtRmqCj5/nvWC93ABAzrXwMZComXUyo QjsNKl/FRMafUgg54lS7fz8siEp0IMHuA9lvhCb+ygSHnTQ0BxLqH3tI8VcoYRRUlm3w B3dx/dGVu+aLbnYQ+E4DEY1CHtc1zCkRURFRO/670DieTVwWxVuzt5aOQIKcxw1gqfNT PA1MnblqmwF22DumPYvlyqokfa7e4z29tEJG0Kc4i6BFt0qUp5MnOtCUyzbRUDGS2M4l md0Q== X-Gm-Message-State: AOAM532nyqZtY9sV4bojJ9GhQL+S+SY6MJs8Lhb+eatT3dCYWgm1SMoY FQGfnoytRHE1dwx+OR7LPqA+QX3Md4a2fw== X-Google-Smtp-Source: ABdhPJwK0duZXj+BCNxn9JZdbdeDmvQ63750ZMPYt67pEWGHkI3wWzwCSdyJAylP1T9zSDZAxZ7ZYQ== X-Received: by 2002:a7b:ca55:: with SMTP id m21mr7396023wml.178.1638369131273; Wed, 01 Dec 2021 06:32:11 -0800 (PST) Received: from localhost ([83.219.56.252]) by smtp.gmail.com with ESMTPSA id w4sm20382180wrs.88.2021.12.01.06.32.10 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Wed, 01 Dec 2021 06:32:10 -0800 (PST) From: Jan Vrany To: gdb-patches@sourceware.org Cc: Jan Vrany , lsix@lancelotsix.com, brobecker@adacore.com, pedro@palves.net Subject: [PATCH v3 2/2] ppc: recognize all program traps Date: Wed, 1 Dec 2021 14:30:05 +0000 Message-Id: <20211201143005.3690440-3-jan.vrany@labware.com> X-Mailer: git-send-email 2.30.2 In-Reply-To: <20211124130926.2412617-1-jan.vrany@labware.com> References: <20211124130926.2412617-1-jan.vrany@labware.com> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit X-Spam-Status: No, score=-11.6 required=5.0 tests=BAYES_00, FREEMAIL_ENVFROM_END_DIGIT, FREEMAIL_FORGED_FROMDOMAIN, FREEMAIL_FROM, GIT_PATCH_0, HEADER_FROM_DIFFERENT_DOMAINS, KAM_DMARC_STATUS, KAM_SHORT, RCVD_IN_DNSWL_NONE, RCVD_IN_MSPIKE_H2, SPF_HELO_NONE, SPF_PASS, TXREP autolearn=ham autolearn_force=no version=3.4.4 X-Spam-Checker-Version: SpamAssassin 3.4.4 (2020-01-24) on server2.sourceware.org X-BeenThere: gdb-patches@sourceware.org X-Mailman-Version: 2.1.29 Precedence: list List-Id: Gdb-patches mailing list List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , X-List-Received-Date: Wed, 01 Dec 2021 14:32:14 -0000 Permanent program breakpoints (ones inserted into the code) other than the one GDB uses for POWER (0x7fe00008) did not result in stop but caused GDB to loop infinitely. This was because GDB did not recognize trap instructions other than "trap". For example, "tw 12, 4, 4" was not be recognized, causing GDB to loop forever. This commit fixes this by providing POWER specific hook (gdbarch_program_breakpoint_here_p) recognizing all tw, twi, td and tdi instructions. Tested on Linux on PowerPC e500 and on QEMU PPC64le. --- gdb/rs6000-tdep.c | 65 ++++++++++++++++++++++ gdb/testsuite/gdb.arch/powerpc-trap.exp | 72 +++++++++++++++++++++++++ gdb/testsuite/gdb.arch/powerpc-trap.s | 31 +++++++++++ gdb/testsuite/gdb.arch/powerpc64-trap.s | 33 ++++++++++++ 4 files changed, 201 insertions(+) create mode 100644 gdb/testsuite/gdb.arch/powerpc-trap.exp create mode 100644 gdb/testsuite/gdb.arch/powerpc-trap.s create mode 100644 gdb/testsuite/gdb.arch/powerpc64-trap.s diff --git a/gdb/rs6000-tdep.c b/gdb/rs6000-tdep.c index 43880fa4426..ce98dc2f884 100644 --- a/gdb/rs6000-tdep.c +++ b/gdb/rs6000-tdep.c @@ -6247,6 +6247,69 @@ ppc_process_record (struct gdbarch *gdbarch, struct regcache *regcache, return 0; } +/* Used for matching tw, twi, td and tdi instructions for POWER. */ + +static constexpr uint32_t TX_INSN_MASK = 0xFC0007FF; +static constexpr uint32_t TW_INSN = 0x7C000008; +static constexpr uint32_t TD_INSN = 0x7C000088; + +static constexpr uint32_t TXI_INSN_MASK = 0xFC000000; +static constexpr uint32_t TWI_INSN = 0x0C000000; +static constexpr uint32_t TDI_INSN = 0x08000000; + +static inline bool +is_tw_insn (uint32_t insn) +{ + return (insn & TX_INSN_MASK) == TW_INSN; +} + +static inline bool +is_twi_insn (uint32_t insn) +{ + return (insn & TXI_INSN_MASK) == TWI_INSN; +} + +static inline bool +is_td_insn (uint32_t insn) +{ + return (insn & TX_INSN_MASK) == TD_INSN; +} + +static inline bool +is_tdi_insn (uint32_t insn) +{ + return (insn & TXI_INSN_MASK) == TDI_INSN; +} + +/* Implementation of gdbarch_program_breakpoint_here_p for POWER. */ + +static bool +rs6000_program_breakpoint_here_p (gdbarch *gdbarch, CORE_ADDR address) +{ + gdb_byte target_mem[PPC_INSN_SIZE]; + + /* Enable the automatic memory restoration from breakpoints while + we read the memory. Otherwise we may find temporary breakpoints, ones + inserted by GDB, and flag them as permanent breakpoints. */ + scoped_restore restore_memory + = make_scoped_restore_show_memory_breakpoints (0); + + if (target_read_memory (address, target_mem, PPC_INSN_SIZE) == 0) + { + uint32_t insn = (uint32_t) extract_unsigned_integer + (target_mem, PPC_INSN_SIZE, gdbarch_byte_order_for_code (gdbarch)); + + /* Check if INSN is a TW, TWI, TD or TDI instruction. There + are multiple choices of such instructions with different registers + and / or immediate values but they all cause a break. */ + if (is_tw_insn (insn) || is_twi_insn (insn) || is_td_insn (insn) + || is_tdi_insn (insn)) + return true; + } + + return false; +} + /* Initialize the current architecture based on INFO. If possible, re-use an architecture from ARCHES, which is a list of architectures already created during this debugging session. @@ -7109,6 +7172,8 @@ rs6000_gdbarch_init (struct gdbarch_info info, struct gdbarch_list *arches) rs6000_breakpoint::kind_from_pc); set_gdbarch_sw_breakpoint_from_kind (gdbarch, rs6000_breakpoint::bp_from_kind); + set_gdbarch_program_breakpoint_here_p (gdbarch, + rs6000_program_breakpoint_here_p); /* The value of symbols of type N_SO and N_FUN maybe null when it shouldn't be. */ diff --git a/gdb/testsuite/gdb.arch/powerpc-trap.exp b/gdb/testsuite/gdb.arch/powerpc-trap.exp new file mode 100644 index 00000000000..9cdbbf0cc30 --- /dev/null +++ b/gdb/testsuite/gdb.arch/powerpc-trap.exp @@ -0,0 +1,72 @@ +# Copyright 2021 Free Software Foundation, Inc. +# +# This program is free software; you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation; either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see . +# +# This file is part of the gdb testsuite. + +# Test if GDB stops at various trap instructions inserted into +# the code. + +if { [istarget powerpc-*] } { + standard_testfile powerpc-trap.s + # Number of expected SIGTRAP's to get. This needs to be kept in sync + # with the source file powerpc-trap.s + set expected_traps 3 +} elseif {[istarget powerpc64*] } { + standard_testfile powerpc64-trap.s + # Number of expected SIGTRAP's to get. This needs to be kept in sync + # with the source file powerpc64-trap.s + set expected_traps 5 +} else { + verbose "Skipping ${gdb_test_file_name}." + return +} + +if {[prepare_for_testing "failed to prepare" ${testfile} ${srcfile}]} { + return -1 +} + +if {![runto_main]} { + untested "could not run to main" + return -1 +} + +set keep_going 1 +set count 0 + +while {$keep_going} { + set keep_going 0 + + # Continue to next program breakpoint instruction. + gdb_test_multiple "continue" "trap instruction $count causes SIGTRAP" { + -re "Program received signal SIGTRAP, Trace/breakpoint trap.*$gdb_prompt $" { + pass $gdb_test_name + + # Advance PC to next instruction + gdb_test "set \$pc = \$pc + 4" "" "advance past trap instruction $count" + + incr count + if {$count < $expected_traps} { + set keep_going 1 + } + } + } +} + +# Verify we stopped at the expected number of SIGTRAP's. +gdb_assert {$count == $expected_traps} "all trap instructions triggered" + +# One last continue to reach the end of the test, to make sure we don't get +# another SIGTRAP. +gdb_test "continue" "exited normally.*" "continue to end" diff --git a/gdb/testsuite/gdb.arch/powerpc-trap.s b/gdb/testsuite/gdb.arch/powerpc-trap.s new file mode 100644 index 00000000000..b03176f747e --- /dev/null +++ b/gdb/testsuite/gdb.arch/powerpc-trap.s @@ -0,0 +1,31 @@ +/* This testcase is part of GDB, the GNU debugger. + + Copyright 2021 Free Software Foundation, Inc. + + This program is free software; you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation; either version 3 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . */ + +/* To test if GDB stops at various trap instructions inserted into + the code. */ + +.global main +.type main,function +main: + ori 0, 0, 0 + trap + tw 12, 2, 2 + twi 31, 3, 3 + ori 0, 0, 0 + li 3, 0 + blr + diff --git a/gdb/testsuite/gdb.arch/powerpc64-trap.s b/gdb/testsuite/gdb.arch/powerpc64-trap.s new file mode 100644 index 00000000000..2272b42edc2 --- /dev/null +++ b/gdb/testsuite/gdb.arch/powerpc64-trap.s @@ -0,0 +1,33 @@ +/* This testcase is part of GDB, the GNU debugger. + + Copyright 2021 Free Software Foundation, Inc. + + This program is free software; you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation; either version 3 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . */ + +/* To test if GDB stops at various trap instructions inserted into + the code. */ + +.global main +.type main,function +main: + ori 0, 0, 0 + trap + tw 12, 2, 2 + twi 31, 3, 3 + td 12, 2, 2 + tdi 31, 3, 3 + ori 0, 0, 0 + li 3, 0 + blr + -- 2.30.2