[12/14] add linux_infcall_mmap
Commit Message
From: Jan Kratochvil <jan.kratochvil@redhat.com>
This implements the new gdbarch "infcall_mmap" method for Linux.
2014-05-16 Jan Kratochvil <jan.kratochvil@redhat.com>
* linux-tdep.c (linux_infcall_mmap): New function.
(linux_init_abi): Add it to gdbarch.
---
gdb/ChangeLog | 5 +++++
gdb/linux-tdep.c | 46 ++++++++++++++++++++++++++++++++++++++++++++++
2 files changed, 51 insertions(+)
@@ -32,6 +32,9 @@
#include "cli/cli-utils.h"
#include "arch-utils.h"
#include "gdb_obstack.h"
+#include "objfiles.h"
+#include "infcall.h"
+#include <sys/mman.h>
#include <ctype.h>
@@ -1766,6 +1769,48 @@ linux_gdb_signal_to_target (struct gdbarch *gdbarch,
return -1;
}
+/* See gdbarch.sh 'infcall_mmap'. */
+
+static CORE_ADDR
+linux_infcall_mmap (CORE_ADDR size, unsigned prot)
+{
+ struct objfile *objf;
+ /* Do there still exist any Linux systems without "mmap64"?
+ "mmap" uses 64-bit off_t on x86_64 and 32-bit off_t on i386 and x32. */
+ struct value *mmap_val = find_function_in_inferior ("mmap64", &objf);
+ struct value *addr_val;
+ struct gdbarch *gdbarch = get_objfile_arch (objf);
+ CORE_ADDR retval;
+ enum
+ {
+ ARG_ADDR, ARG_LENGTH, ARG_PROT, ARG_FLAGS, ARG_FD, ARG_OFFSET, ARG_MAX
+ };
+ struct value *arg[ARG_MAX];
+
+ arg[ARG_ADDR] = value_from_pointer (builtin_type (gdbarch)->builtin_data_ptr,
+ 0);
+ /* Assuming sizeof (unsigned long) == sizeof (size_t). */
+ arg[ARG_LENGTH] = value_from_ulongest
+ (builtin_type (gdbarch)->builtin_unsigned_long, size);
+ gdb_assert ((prot & ~7) == 0);
+ arg[ARG_PROT] = value_from_longest (builtin_type (gdbarch)->builtin_int,
+ 0
+ | ((prot & 4) != 0 ? PROT_READ : 0)
+ | ((prot & 2) != 0 ? PROT_WRITE : 0)
+ | ((prot & 1) != 0 ? PROT_EXEC : 0));
+ arg[ARG_FLAGS] = value_from_longest (builtin_type (gdbarch)->builtin_int,
+ MAP_PRIVATE | MAP_ANONYMOUS);
+ arg[ARG_FD] = value_from_longest (builtin_type (gdbarch)->builtin_int, -1);
+ arg[ARG_OFFSET] = value_from_longest (builtin_type (gdbarch)->builtin_int64,
+ 0);
+ addr_val = call_function_by_hand (mmap_val, ARG_MAX, arg);
+ retval = value_as_address (addr_val);
+ if (retval == (CORE_ADDR) -1)
+ error (_("Failed inferior mmap call for %s bytes, errno is changed."),
+ pulongest (size));
+ return retval;
+}
+
/* To be called from the various GDB_OSABI_LINUX handlers for the
various GNU/Linux architectures and machine types. */
@@ -1783,6 +1828,7 @@ linux_init_abi (struct gdbarch_info info, struct gdbarch *gdbarch)
linux_gdb_signal_from_target);
set_gdbarch_gdb_signal_to_target (gdbarch,
linux_gdb_signal_to_target);
+ set_gdbarch_infcall_mmap (gdbarch, linux_infcall_mmap);
}
/* Provide a prototype to silence -Wmissing-prototypes. */