summaryrefslogtreecommitdiff
path: root/gdb/testsuite
diff options
context:
space:
mode:
authorAndreas Arnez <arnez@linux.vnet.ibm.com>2016-09-09 19:59:53 +0200
committerAndreas Arnez <arnez@linux.vnet.ibm.com>2016-09-09 19:59:53 +0200
commite1b2624a08fae1f669d879946d5041945b4dc248 (patch)
tree2b3634606789cdb6b21fd1a0fe5e0e6b666f9a82 /gdb/testsuite
parent3569342c148dd1cb4b2e1bdafe64a9e3a3701813 (diff)
downloadbinutils-gdb-e1b2624a08fae1f669d879946d5041945b4dc248.tar.gz
Pass HWCAP to ifunc resolver
On various GNU Elf architectures, including AArch64, ARM, s390/s390x, ppc32/64, and sparc32/64, the dynamic loader passes HWCAP as a parameter to each ifunc resolver. Currently there is an open glibc Bugzilla that requests this to be generalized to all architectures: https://sourceware.org/bugzilla/show_bug.cgi?id=19766 And various ifunc resolvers already rely on receiving HWCAP. Currently GDB always calls an ifunc resolver without any arguments; thus the resolver may receive garbage, and based on that, the resolver may decide to return a function that is not suited for the given platform. This patch always passes HWCAP to ifunc resolvers, even on systems where the dynamic loader currently behaves otherwise. The rationale is that (1) the dynamic loader may get adjusted on those systems as well in the future; (2) passing an unused argument should not cause a problem with existing resolvers; and (3) the logic is much simpler without such a distinction. gdb/ChangeLog: * elfread.c (auxv.h): New include. (elf_gnu_ifunc_resolve_addr): Pass HWCAP to ifunc resolver. gdb/testsuite/ChangeLog: * gdb.base/gnu-ifunc-lib.c (resolver_hwcap): New external variable declaration. (gnu_ifunc): Add parameter hwcap. Store it in resolver_hwcap. * gdb.base/gnu-ifunc.c (resolver_hwcap): New global variable. * gdb.base/gnu-ifunc.exp: Add test to verify that the resolver received HWCAP as its argument.
Diffstat (limited to 'gdb/testsuite')
-rw-r--r--gdb/testsuite/ChangeLog9
-rw-r--r--gdb/testsuite/gdb.base/gnu-ifunc-lib.c4
-rw-r--r--gdb/testsuite/gdb.base/gnu-ifunc.c4
-rw-r--r--gdb/testsuite/gdb.base/gnu-ifunc.exp15
4 files changed, 31 insertions, 1 deletions
diff --git a/gdb/testsuite/ChangeLog b/gdb/testsuite/ChangeLog
index 958ec27b626..783e30fedb0 100644
--- a/gdb/testsuite/ChangeLog
+++ b/gdb/testsuite/ChangeLog
@@ -1,3 +1,12 @@
+2016-09-09 Andreas Arnez <arnez@linux.vnet.ibm.com>
+
+ * gdb.base/gnu-ifunc-lib.c (resolver_hwcap): New external
+ variable declaration.
+ (gnu_ifunc): Add parameter hwcap. Store it in resolver_hwcap.
+ * gdb.base/gnu-ifunc.c (resolver_hwcap): New global variable.
+ * gdb.base/gnu-ifunc.exp: Add test to verify that the resolver
+ received HWCAP as its argument.
+
2016-09-06 Pedro Alves <palves@redhat.com>
* gdb.base/new-ui-pending-input.c: New file.
diff --git a/gdb/testsuite/gdb.base/gnu-ifunc-lib.c b/gdb/testsuite/gdb.base/gnu-ifunc-lib.c
index 8a55f600396..0c0aeef364f 100644
--- a/gdb/testsuite/gdb.base/gnu-ifunc-lib.c
+++ b/gdb/testsuite/gdb.base/gnu-ifunc-lib.c
@@ -16,6 +16,7 @@
along with this program. If not, see <http://www.gnu.org/licenses/>. */
extern volatile int gnu_ifunc_initialized;
+extern volatile unsigned long resolver_hwcap;
extern int init_stub (int arg);
extern int final (int arg);
@@ -24,8 +25,9 @@ typedef int (*final_t) (int arg);
asm (".type gnu_ifunc, %gnu_indirect_function");
final_t
-gnu_ifunc (void)
+gnu_ifunc (unsigned long hwcap)
{
+ resolver_hwcap = hwcap;
if (! gnu_ifunc_initialized)
return init_stub;
else
diff --git a/gdb/testsuite/gdb.base/gnu-ifunc.c b/gdb/testsuite/gdb.base/gnu-ifunc.c
index c68866ec84d..77dea30d96b 100644
--- a/gdb/testsuite/gdb.base/gnu-ifunc.c
+++ b/gdb/testsuite/gdb.base/gnu-ifunc.c
@@ -35,6 +35,10 @@ final (int arg)
volatile int gnu_ifunc_initialized;
+/* This stores the argument received by the ifunc resolver. */
+
+volatile unsigned long resolver_hwcap = -1;
+
static void
gnu_ifunc_pre (void)
{
diff --git a/gdb/testsuite/gdb.base/gnu-ifunc.exp b/gdb/testsuite/gdb.base/gnu-ifunc.exp
index 097e48a9a0c..3b2775b8c76 100644
--- a/gdb/testsuite/gdb.base/gnu-ifunc.exp
+++ b/gdb/testsuite/gdb.base/gnu-ifunc.exp
@@ -76,6 +76,21 @@ gdb_continue_to_breakpoint "break-at-call" ".*break-at-call.*"
gdb_test "p gnu_ifunc (3)" " = 4"
+# Test that the resolver received its argument.
+
+set actual_hwcap "0x0"
+set test "info auxv"
+gdb_test_multiple $test $test {
+ -re "\r\n\\d+\\s+AT_HWCAP\[^\r\n\]+($hex)\r\n.*$gdb_prompt $" {
+ set actual_hwcap $expect_out(1,string)
+ }
+ -re ".*$gdb_prompt $" {
+ pass "$test (no HWCAP)"
+ }
+}
+
+gdb_test "p/x resolver_hwcap" "= $actual_hwcap" "resolver received HWCAP"
+
# Test GDB will skip the gnu_ifunc resolver on first call.
gdb_test "step" "\r\nfinal .*"