mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2024-11-30 21:44:19 +08:00
6c95b8df7f
Stan Shebs <stan@codesourcery.com> Add base multi-executable/process support to GDB. gdb/ * Makefile.in (SFILES): Add progspace.c. (COMMON_OBS): Add progspace.o. * progspace.h: New. * progspace.c: New. * breakpoint.h (struct bp_target_info) <placed_address_space>: New field. (struct bp_location) <pspace>: New field. (struct breakpoint) <pspace>: New field. (bpstat_stop_status, breakpoint_here_p) (moribund_breakpoint_here_p, breakpoint_inserted_here_p) (regular_breakpoint_inserted_here_p) (software_breakpoint_inserted_here_p, breakpoint_thread_match) (set_default_breakpoint): Adjust prototypes. (remove_breakpoints_pid, breakpoint_program_space_exit): Declare. (insert_single_step_breakpoint, deprecated_insert_raw_breakpoint): Adjust prototypes. * breakpoint.c (executing_startup): Delete. (default_breakpoint_sspace): New. (breakpoint_restore_shadows): Skip if the address space doesn't match. (update_watchpoint): Record the frame's program space in the breakpoint location. (insert_bp_location): Record the address space in target_info. Adjust to pass the symbol space to solib_name_from_address. (breakpoint_program_space_exit): New. (insert_breakpoint_locations): Switch the symbol space and thread when inserting breakpoints. Don't insert breakpoints in a vfork parent waiting for vfork done if we're not attached to the vfork child. (remove_breakpoints_pid): New. (reattach_breakpoints): Switch to a thread of PID. Ignore breakpoints of other symbol spaces. (create_internal_breakpoint): Store the symbol space in the sal. (create_longjmp_master_breakpoint): Iterate over all symbol spaces. (update_breakpoints_after_exec): Ignore breakpoints for other symbol spaces. (remove_breakpoint): Rename to ... (remove_breakpoint_1): ... this. Pass the breakpoints symbol space to solib_name_from_address. (remove_breakpoint): New. (mark_breakpoints_out): Ignore breakpoints from other symbol spaces. (breakpoint_init_inferior): Ditto. (breakpoint_here_p): Add an address space argument and adjust to use breakpoint_address_match. (moribund_breakpoint_here_p): Ditto. (regular_breakpoint_inserted_here_p): Ditto. (breakpoint_inserted_here_p): Ditto. (software_breakpoint_inserted_here_p): Ditto. (breakpoint_thread_match): Ditto. (bpstat_check_location): Ditto. (bpstat_stop_status): Ditto. (print_breakpoint_location): If there's a location to print, switch the current symbol space. (print_one_breakpoint_location): Add `allflag' argument. (print_one_breakpoint): Ditto. Adjust. (do_captured_breakpoint_query): Adjust. (breakpoint_1): Adjust. (breakpoint_has_pc): Also match the symbol space. (describe_other_breakpoints): Add a symbol space argument and adjust. (set_default_breakpoint): Add a symbol space argument. Set default_breakpoint_sspace. (breakpoint_address_match): New. (check_duplicates_for): Add an address space argument, and adjust. (set_raw_breakpoint): Record the symbol space in the location and in the breakpoint. (set_longjmp_breakpoint): Skip longjmp master breakpoints from other symbol spaces. (remove_thread_event_breakpoints, remove_solib_event_breakpoints) (disable_breakpoints_in_shlibs): Skip breakpoints from other symbol spaces. (disable_breakpoints_in_unloaded_shlib): Match symbol spaces. (create_catchpoint): Set the symbol space in the sal. (disable_breakpoints_before_startup): Skip breakpoints from other symbol spaces. Set executing_startup in the current symbol space. (enable_breakpoints_after_startup): Clear executing_startup in the current symbol space. Skip breakpoints from other symbol spaces. (clone_momentary_breakpoint): Also copy the symbol space. (add_location_to_breakpoint): Set the location's symbol space. (bp_loc_is_permanent): Switch thread and symbol space. (create_breakpoint): Adjust. (expand_line_sal_maybe): Expand comment to mention symbol spaces. Switch thread and symbol space when reading memory. (parse_breakpoint_sals): Set the symbol space in the sal. (break_command_really): Ditto. (skip_prologue_sal): Switch and space. (resolve_sal_pc): Ditto. (watch_command_1): Record the symbol space in the sal. (create_ada_exception_breakpoint): Adjust. (clear_command): Adjust. Match symbol spaces. (update_global_location_list): Use breakpoint_address_match. (breakpoint_re_set_one): Switch thread and space. (breakpoint_re_set): Save symbol space. (breakpoint_re_set_thread): Also reset the symbol space. (deprecated_insert_raw_breakpoint): Add an address space argument. Adjust. (insert_single_step_breakpoint): Ditto. (single_step_breakpoint_inserted_here_p): Ditto. (clear_syscall_counts): New. (_initialize_breakpoint): Install it as inferior_exit observer. * exec.h: Include "progspace.h". (exec_bfd, exec_bfd_mtime): New defines. (exec_close): Declare. * exec.c: Include "gdbthread.h" and "progspace.h". (exec_bfd, exec_bfd_mtime, current_target_sections_1): Delete. (using_exec_ops): New. (exec_close_1): Rename to exec_close, and make public. (exec_close): Rename to exec_close_1, and adjust all callers. Add description. Remove target sections and close executables from all program spaces. (exec_file_attach): Add comment. (add_target_sections): Check on `using_exec_ops' to check if the target should be pushed. (remove_target_sections): Only unpush the target if there are no more target sections in any symbol space. * gdbcore.h: Include "exec.h". (exec_bfd, exec_bfd_mtime): Remove declarations. * frame.h (get_frame_program_space, get_frame_address_space) (frame_unwind_program_space): Declare. * frame.c (struct frame_info) <pspace, aspace>: New fields. (create_sentinel_frame): Add program space argument. Set the pspace and aspace fields of the frame object. (get_current_frame, create_new_frame): Adjust. (get_frame_program_space): New. (frame_unwind_program_space): New. (get_frame_address_space): New. * stack.c (print_frame_info): Adjust. (print_frame): Use the frame's program space. * gdbthread.h (any_live_thread_of_process): Declare. * thread.c (any_live_thread_of_process): New. (switch_to_thread): Switch the program space as well. (restore_selected_frame): Don't warn if trying to restore frame level 0. * inferior.h: Include "progspace.h". (detach_fork): Declare. (struct inferior) <removable, aspace, pspace> <vfork_parent, vfork_child, pending_detach> <waiting_for_vfork_done>: New fields. <terminal_info>: Remove field. <data, num_data>: New fields. (register_inferior_data, register_inferior_data_with_cleanup) (clear_inferior_data, set_inferior_data, inferior_data): Declare. (exit_inferior, exit_inferior_silent, exit_inferior_num_silent) (inferior_appeared): Declare. (find_inferior_pid): Typo. (find_inferior_id, find_inferior_for_program_space): Declare. (set_current_inferior, save_current_inferior, prune_inferiors) (number_of_inferiors): Declare. (inferior_list): Declare. * inferior.c: Include "gdbcore.h" and "symfile.h". (inferior_list): Make public. (delete_inferior_1): Always delete thread silently. (find_inferior_id): Make public. (current_inferior_): New. (current_inferior): Use it. (set_current_inferior): New. (restore_inferior): New. (save_current_inferior): New. (free_inferior): Free the per-inferior data. (add_inferior_silent): Allocate per-inferior data. Call inferior_appeared. (delete_threads_of_inferior): New. (delete_inferior_1): Adjust interface to take an inferior pointer. (delete_inferior): Adjust. (delete_inferior_silent): Adjust. (exit_inferior_1): New. (exit_inferior): New. (exit_inferior_silent): New. (exit_inferior_num_silent): New. (detach_inferior): Adjust. (inferior_appeared): New. (discard_all_inferiors): Adjust. (find_inferior_id): Make public. Assert pid is not zero. (find_inferior_for_program_space): New. (have_inferiors): Check if we have any inferior with pid not zero. (have_live_inferiors): Go over all pushed targets looking for process_stratum. (prune_inferiors): New. (number_of_inferiors): New. (print_inferior): Add executable column. Print vfork parent/child relationships. (inferior_command): Adjust to cope with not running inferiors. (remove_inferior_command): New. (add_inferior_command): New. (clone_inferior_command): New. (struct inferior_data): New. (struct inferior_data_registration): New. (struct inferior_data_registry): New. (inferior_data_registry): New. (register_inferior_data_with_cleanup): New. (register_inferior_data): New. (inferior_alloc_data): New. (inferior_free_data): New. (clear_inferior_data): New. (set_inferior_data): New. (inferior_data): New. (initialize_inferiors): New. (_initialize_inferiors): Register "add-inferior", "remove-inferior" and "clone-inferior" commands. * objfiles.h: Include "progspace.h". (struct objfile) <pspace>: New field. (symfile_objfile, object_files): Don't declare. (ALL_PSPACE_OBJFILES): New. (ALL_PSPACE_OBJFILES_SAFE): New. (ALL_OBJFILES, ALL_OBJFILES_SAFE): Adjust. (ALL_PSPACE_SYMTABS): New. (ALL_PRIMARY_SYMTABS): Adjust. (ALL_PSPACE_PRIMARY_SYMTABS): New. (ALL_PSYMTABS): Adjust. (ALL_PSPACE_PSYMTABS): New. * objfiles.c (object_files, symfile_objfile): Delete. (struct objfile_sspace_info): New. (objfiles_pspace_data): New. (objfiles_pspace_data_cleanup): New. (get_objfile_pspace_data): New. (objfiles_changed_p): Delete. (allocate_objfile): Set the objfile's program space. Adjust to reference objfiles_changed_p in pspace data. (free_objfile): Adjust to reference objfiles_changed_p in pspace data. (objfile_relocate): Ditto. (update_section_map): Add pspace argument. Adjust to iterate over objfiles in the passed in pspace. (find_pc_section): Delete sections and num_sections statics. Adjust to refer to program space's objfiles_changed_p. Adjust to refer to sections and num_sections store in the objfile's pspace data. (objfiles_changed): Adjust to reference objfiles_changed_p in pspace data. (_initialize_objfiles): New. * linespec.c (decode_all_digits, decode_dollar): Set the sal's program space. * source.c (current_source_pspace): New. (get_current_source_symtab_and_line): Set the sal's program space. (set_current_source_symtab_and_line): Set current_source_pspace. (select_source_symtab): Ditto. Use ALL_OBJFILES. (forget_cached_source_info): Iterate over all program spaces. * symfile.c (clear_symtab_users): Adjust. * symmisc.c (print_symbol_bcache_statistics): Iterate over all program spaces. (print_objfile_statistics): Ditto. (maintenance_print_msymbols): Ditto. (maintenance_print_objfiles): Ditto. (maintenance_info_symtabs): Ditto. (maintenance_info_psymtabs): Ditto. * symtab.h (SYMTAB_PSPACE): New. (struct symtab_and_line) <pspace>: New field. * symtab.c (init_sal): Clear the sal's program space. (find_pc_sect_symtab): Set the sal's program space. Switch thread and space. (append_expanded_sal): Add program space argument. Iterate over all program spaces. (expand_line_sal): Iterate over all program spaces. Switch program space. * target.h (enum target_waitkind) <TARGET_WAITKIND_VFORK_DONE>: New. (struct target_ops) <to_thread_address_space>: New field. (target_thread_address_space): Define. * target.c (target_detach): Only remove breakpoints from the inferior we're detaching. (target_thread_address_space): New. * defs.h (initialize_progspace): Declare. * top.c (gdb_init): Call it. * solist.h (struct so_list) <sspace>: New field. * solib.h (struct program_space): Forward declare. (solib_name_from_address): Adjust prototype. * solib.c (so_list_head): Replace with a macro referencing the program space. (update_solib_list): Set the so's program space. (solib_name_from_address): Add a program space argument and adjust. * solib-svr4.c (struct svr4_info) <pid>: Delete field. <interp_text_sect_low, interp_text_sect_high, interp_plt_sect_low> <interp_plt_sect_high>: New fields. (svr4_info_p, svr4_info): Delete. (solib_svr4_sspace_data): New. (get_svr4_info): Rewrite. (svr4_sspace_data_cleanup): New. (open_symbol_file_object): Adjust. (svr4_default_sos): Adjust. (svr4_fetch_objfile_link_map): Adjust. (interp_text_sect_low, interp_text_sect_high, interp_plt_sect_low) (interp_plt_sect_high): Delete. (svr4_in_dynsym_resolve_code): Adjust. (enable_break): Adjust. (svr4_clear_solib): Revert bit that removed the svr4_info here, and reinstate clearing debug_base, debug_loader_offset_p, debug_loader_offset and debug_loader_name. (_initialize_svr4_solib): Register solib_svr4_pspace_data. Don't install an inferior_exit observer anymore. * printcmd.c (struct display) <pspace>: New field. (display_command): Set the display's sspace. (do_one_display): Match the display's sspace. (display_uses_solib_p): Ditto. * linux-fork.c (detach_fork): Moved to infrun.c. (_initialize_linux_fork): Moved "detach-on-fork" command to infrun.c. * infrun.c (detach_fork): Moved from linux-fork.c. (proceed_after_vfork_done): New. (handle_vfork_child_exec_or_exit): New. (follow_exec_mode_replace, follow_exec_mode_keep) (follow_exec_mode_names, follow_exec_mode_string) (show_follow_exec_mode_string): New. (follow_exec): New. Reinstate the mark_breakpoints_out call. Remove shared libraries before attaching new executable. If user wants to keep the inferior, keep it. (displaced_step_fixup): Adjust to pass an address space to the breakpoints module. (resume): Ditto. (clear_proceed_status): In all-stop mode, always clear the proceed status of all threads. (prepare_to_proceed): Adjust to pass an address space to the breakpoints module. (proceed): Ditto. (adjust_pc_after_break): Ditto. (handle_inferior_event): When handling a process exit, switch the program space to the inferior's that had exited. Call handle_vfork_child_exec_or_exit. Adjust to pass an address space to the breakpoints module. In non-stop mode, when following a fork and detach-fork is off, also resume the other branch. Handle TARGET_WAITKIND_VFORK_DONE. Set the program space in sals. (normal_stop): Prune inferiors. (_initialize_infrun): Install the new "follow-exec-mode" command. "detach-on-fork" moved here. * regcache.h (get_regcache_aspace): Declare. * regcache.c (struct regcache) <aspace>: New field. (regcache_xmalloc): Clear the aspace. (get_regcache_aspace): New. (regcache_cpy): Copy the aspace field. (regcache_cpy_no_passthrough): Ditto. (get_thread_regcache): Fetch the thread's address space from the target, and store it in the regcache. * infcall.c (call_function_by_hand): Set the sal's pspace. * arch-utils.c (default_has_shared_address_space): New. * arch-utils.h (default_has_shared_address_space): Declare. * gdbarch.sh (has_shared_address_space): New. * gdbarch.h, gdbarch.c: Regenerate. * linux-tdep.c: Include auxv.h, target.h, elf/common.h. (linux_has_shared_address_space): New. (_initialize_linux_tdep): Declare. * arm-tdep.c (arm_software_single_step): Pass the frame's address space to insert_single_step_breakpoint. * arm-linux-tdep.c (arm_linux_software_single_step): Pass the frame's pspace to breakpoint functions. * cris-tdep.c (crisv32_single_step_through_delay): Ditto. (cris_software_single_step): Ditto. * mips-tdep.c (deal_with_atomic_sequence): Add frame argument. Pass the frame's pspace to breakpoint functions. (mips_software_single_step): Adjust. (mips_single_step_through_delay): Adjust. * rs6000-aix-tdep.c (rs6000_software_single_step): Adjust. * rs6000-tdep.c (ppc_deal_with_atomic_sequence): Adjust. * solib-irix.c (enable_break): Adjust to pass the current frame's address space to breakpoint functions. * sparc-tdep.c (sparc_software_single_step): Ditto. * spu-tdep.c (spu_software_single_step): Ditto. * alpha-tdep.c (alpha_software_single_step): Ditto. * record.c (record_wait): Adjust to pass an address space to the breakpoints module. * fork-child.c (fork_inferior): Set the new inferior's program and address spaces. * inf-ptrace.c (inf_ptrace_follow_fork): Copy the parent's program and address spaces. (inf_ptrace_attach): Set the inferior's program and address spaces. * linux-nat.c: Include "solib.h". (linux_child_follow_fork): Manage parent and child's program and address spaces. Clone the parent's program space if necessary. Don't wait for the vfork to be done here. Refuse to resume if following the vfork parent while leaving the child stopped. (resume_callback): Don't resume a vfork parent. (linux_nat_resume): Also check for pending events in the lp->waitstatus field. (linux_handle_extended_wait): Report TARGET_WAITKIND_VFORK_DONE events to the core. (stop_wait_callback): Don't wait for SIGSTOP on vfork parents. (cancel_breakpoint): Adjust. * linux-thread-db.c (thread_db_wait): Don't remove thread event breakpoints here. (thread_db_mourn_inferior): Don't mark breakpoints out here. Remove thread event breakpoints after mourning. * corelow.c: Include progspace.h. (core_open): Set the inferior's program and address spaces. * remote.c (remote_add_inferior): Set the new inferior's program and address spaces. (remote_start_remote): Update address spaces. (extended_remote_create_inferior_1): Don't init the thread list if we already debugging other inferiors. * darwin-nat.c (darwin_attach): Set the new inferior's program and address spaces. * gnu-nat.c (gnu_attach): Ditto. * go32-nat.c (go32_create_inferior): Ditto. * inf-ttrace.c (inf_ttrace_follow_fork, inf_ttrace_attach): Ditto. * monitor.c (monitor_open): Ditto. * nto-procfs.c (procfs_attach, procfs_create_inferior): Ditto. * procfs.c (do_attach): Ditto. * windows-nat.c (do_initial_windows_stuff): Ditto. * inflow.c (inferior_process_group) (terminal_init_inferior_with_pgrp, terminal_inferior, (terminal_ours_1, inflow_inferior_exit, copy_terminal_info) (child_terminal_info, new_tty_postfork, set_sigint_trap): Adjust to use per-inferior data instead of inferior->terminal_info. (inflow_inferior_data): New. (inflow_new_inferior): Delete. (inflow_inferior_data_cleanup): New. (get_inflow_inferior_data): New. * mi/mi-interp.c (mi_new_inferior): Rename to... (mi_inferior_appeared): ... this. (mi_interpreter_init): Adjust. * tui/tui-disasm.c: Include "progspace.h". (tui_set_disassem_content): Pass an address space to breakpoint_here_p. * NEWS: Mention multi-program debugging support. Mention new commands "add-inferior", "clone-inferior", "remove-inferior", "maint info program-spaces", and new option "set follow-exec-mode". 2009-10-19 Pedro Alves <pedro@codesourcery.com> Stan Shebs <stan@codesourcery.com> gdb/doc/ * observer.texi (new_inferior): Rename to... (inferior_appeared): ... this. 2009-10-19 Pedro Alves <pedro@codesourcery.com> Stan Shebs <stan@codesourcery.com> gdb/testsuite/ * gdb.base/foll-vfork.exp: Adjust to spell out "follow-fork". * gdb.base/foll-exec.exp: Adjust to expect a process id before "Executing new program". * gdb.base/foll-fork.exp: Adjust to spell out "follow-fork". * gdb.base/multi-forks.exp: Ditto. Adjust to the inferior being left listed after having been killed. * gdb.base/attach.exp: Adjust to spell out "symbol-file". * gdb.base/maint.exp: Adjust test. * Makefile.in (ALL_SUBDIRS): Add gdb.multi. * gdb.multi/Makefile.in: New. * gdb.multi/base.exp: New. * gdb.multi/goodbye.c: New. * gdb.multi/hangout.c: New. * gdb.multi/hello.c: New. * gdb.multi/bkpt-multi-exec.c: New. * gdb.multi/bkpt-multi-exec.exp: New. * gdb.multi/crashme.c: New. 2009-10-19 Pedro Alves <pedro@codesourcery.com> Stan Shebs <stan@codesourcery.com> gdb/doc/ * gdb.texinfo (Inferiors): Rename node to ... (Inferiors and Programs): ... this. Mention running multiple programs in the same debug session. <info inferiors>: Mention the new 'Executable' column if "info inferiors". Update examples. Document the "add-inferior", "clone-inferior", "remove-inferior" and "maint info program-spaces" commands. (Process): Rename node to... (Forks): ... this. Document "set|show follow-exec-mode".
888 lines
29 KiB
C
888 lines
29 KiB
C
/* GNU/Linux on ARM target support.
|
|
|
|
Copyright (C) 1999, 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008,
|
|
2009 Free Software Foundation, Inc.
|
|
|
|
This file is part of GDB.
|
|
|
|
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 <http://www.gnu.org/licenses/>. */
|
|
|
|
#include "defs.h"
|
|
#include "target.h"
|
|
#include "value.h"
|
|
#include "gdbtypes.h"
|
|
#include "floatformat.h"
|
|
#include "gdbcore.h"
|
|
#include "frame.h"
|
|
#include "regcache.h"
|
|
#include "doublest.h"
|
|
#include "solib-svr4.h"
|
|
#include "osabi.h"
|
|
#include "regset.h"
|
|
#include "trad-frame.h"
|
|
#include "tramp-frame.h"
|
|
#include "breakpoint.h"
|
|
|
|
#include "arm-tdep.h"
|
|
#include "arm-linux-tdep.h"
|
|
#include "linux-tdep.h"
|
|
#include "glibc-tdep.h"
|
|
#include "arch-utils.h"
|
|
#include "inferior.h"
|
|
#include "gdbthread.h"
|
|
#include "symfile.h"
|
|
|
|
#include "gdb_string.h"
|
|
|
|
extern int arm_apcs_32;
|
|
|
|
/* Under ARM GNU/Linux the traditional way of performing a breakpoint
|
|
is to execute a particular software interrupt, rather than use a
|
|
particular undefined instruction to provoke a trap. Upon exection
|
|
of the software interrupt the kernel stops the inferior with a
|
|
SIGTRAP, and wakes the debugger. */
|
|
|
|
static const char arm_linux_arm_le_breakpoint[] = { 0x01, 0x00, 0x9f, 0xef };
|
|
|
|
static const char arm_linux_arm_be_breakpoint[] = { 0xef, 0x9f, 0x00, 0x01 };
|
|
|
|
/* However, the EABI syscall interface (new in Nov. 2005) does not look at
|
|
the operand of the swi if old-ABI compatibility is disabled. Therefore,
|
|
use an undefined instruction instead. This is supported as of kernel
|
|
version 2.5.70 (May 2003), so should be a safe assumption for EABI
|
|
binaries. */
|
|
|
|
static const char eabi_linux_arm_le_breakpoint[] = { 0xf0, 0x01, 0xf0, 0xe7 };
|
|
|
|
static const char eabi_linux_arm_be_breakpoint[] = { 0xe7, 0xf0, 0x01, 0xf0 };
|
|
|
|
/* All the kernels which support Thumb support using a specific undefined
|
|
instruction for the Thumb breakpoint. */
|
|
|
|
static const char arm_linux_thumb_be_breakpoint[] = {0xde, 0x01};
|
|
|
|
static const char arm_linux_thumb_le_breakpoint[] = {0x01, 0xde};
|
|
|
|
/* Description of the longjmp buffer. */
|
|
#define ARM_LINUX_JB_ELEMENT_SIZE INT_REGISTER_SIZE
|
|
#define ARM_LINUX_JB_PC 21
|
|
|
|
/*
|
|
Dynamic Linking on ARM GNU/Linux
|
|
--------------------------------
|
|
|
|
Note: PLT = procedure linkage table
|
|
GOT = global offset table
|
|
|
|
As much as possible, ELF dynamic linking defers the resolution of
|
|
jump/call addresses until the last minute. The technique used is
|
|
inspired by the i386 ELF design, and is based on the following
|
|
constraints.
|
|
|
|
1) The calling technique should not force a change in the assembly
|
|
code produced for apps; it MAY cause changes in the way assembly
|
|
code is produced for position independent code (i.e. shared
|
|
libraries).
|
|
|
|
2) The technique must be such that all executable areas must not be
|
|
modified; and any modified areas must not be executed.
|
|
|
|
To do this, there are three steps involved in a typical jump:
|
|
|
|
1) in the code
|
|
2) through the PLT
|
|
3) using a pointer from the GOT
|
|
|
|
When the executable or library is first loaded, each GOT entry is
|
|
initialized to point to the code which implements dynamic name
|
|
resolution and code finding. This is normally a function in the
|
|
program interpreter (on ARM GNU/Linux this is usually
|
|
ld-linux.so.2, but it does not have to be). On the first
|
|
invocation, the function is located and the GOT entry is replaced
|
|
with the real function address. Subsequent calls go through steps
|
|
1, 2 and 3 and end up calling the real code.
|
|
|
|
1) In the code:
|
|
|
|
b function_call
|
|
bl function_call
|
|
|
|
This is typical ARM code using the 26 bit relative branch or branch
|
|
and link instructions. The target of the instruction
|
|
(function_call is usually the address of the function to be called.
|
|
In position independent code, the target of the instruction is
|
|
actually an entry in the PLT when calling functions in a shared
|
|
library. Note that this call is identical to a normal function
|
|
call, only the target differs.
|
|
|
|
2) In the PLT:
|
|
|
|
The PLT is a synthetic area, created by the linker. It exists in
|
|
both executables and libraries. It is an array of stubs, one per
|
|
imported function call. It looks like this:
|
|
|
|
PLT[0]:
|
|
str lr, [sp, #-4]! @push the return address (lr)
|
|
ldr lr, [pc, #16] @load from 6 words ahead
|
|
add lr, pc, lr @form an address for GOT[0]
|
|
ldr pc, [lr, #8]! @jump to the contents of that addr
|
|
|
|
The return address (lr) is pushed on the stack and used for
|
|
calculations. The load on the second line loads the lr with
|
|
&GOT[3] - . - 20. The addition on the third leaves:
|
|
|
|
lr = (&GOT[3] - . - 20) + (. + 8)
|
|
lr = (&GOT[3] - 12)
|
|
lr = &GOT[0]
|
|
|
|
On the fourth line, the pc and lr are both updated, so that:
|
|
|
|
pc = GOT[2]
|
|
lr = &GOT[0] + 8
|
|
= &GOT[2]
|
|
|
|
NOTE: PLT[0] borrows an offset .word from PLT[1]. This is a little
|
|
"tight", but allows us to keep all the PLT entries the same size.
|
|
|
|
PLT[n+1]:
|
|
ldr ip, [pc, #4] @load offset from gotoff
|
|
add ip, pc, ip @add the offset to the pc
|
|
ldr pc, [ip] @jump to that address
|
|
gotoff: .word GOT[n+3] - .
|
|
|
|
The load on the first line, gets an offset from the fourth word of
|
|
the PLT entry. The add on the second line makes ip = &GOT[n+3],
|
|
which contains either a pointer to PLT[0] (the fixup trampoline) or
|
|
a pointer to the actual code.
|
|
|
|
3) In the GOT:
|
|
|
|
The GOT contains helper pointers for both code (PLT) fixups and
|
|
data fixups. The first 3 entries of the GOT are special. The next
|
|
M entries (where M is the number of entries in the PLT) belong to
|
|
the PLT fixups. The next D (all remaining) entries belong to
|
|
various data fixups. The actual size of the GOT is 3 + M + D.
|
|
|
|
The GOT is also a synthetic area, created by the linker. It exists
|
|
in both executables and libraries. When the GOT is first
|
|
initialized , all the GOT entries relating to PLT fixups are
|
|
pointing to code back at PLT[0].
|
|
|
|
The special entries in the GOT are:
|
|
|
|
GOT[0] = linked list pointer used by the dynamic loader
|
|
GOT[1] = pointer to the reloc table for this module
|
|
GOT[2] = pointer to the fixup/resolver code
|
|
|
|
The first invocation of function call comes through and uses the
|
|
fixup/resolver code. On the entry to the fixup/resolver code:
|
|
|
|
ip = &GOT[n+3]
|
|
lr = &GOT[2]
|
|
stack[0] = return address (lr) of the function call
|
|
[r0, r1, r2, r3] are still the arguments to the function call
|
|
|
|
This is enough information for the fixup/resolver code to work
|
|
with. Before the fixup/resolver code returns, it actually calls
|
|
the requested function and repairs &GOT[n+3]. */
|
|
|
|
/* The constants below were determined by examining the following files
|
|
in the linux kernel sources:
|
|
|
|
arch/arm/kernel/signal.c
|
|
- see SWI_SYS_SIGRETURN and SWI_SYS_RT_SIGRETURN
|
|
include/asm-arm/unistd.h
|
|
- see __NR_sigreturn, __NR_rt_sigreturn, and __NR_SYSCALL_BASE */
|
|
|
|
#define ARM_LINUX_SIGRETURN_INSTR 0xef900077
|
|
#define ARM_LINUX_RT_SIGRETURN_INSTR 0xef9000ad
|
|
|
|
/* For ARM EABI, the syscall number is not in the SWI instruction
|
|
(instead it is loaded into r7). We recognize the pattern that
|
|
glibc uses... alternatively, we could arrange to do this by
|
|
function name, but they are not always exported. */
|
|
#define ARM_SET_R7_SIGRETURN 0xe3a07077
|
|
#define ARM_SET_R7_RT_SIGRETURN 0xe3a070ad
|
|
#define ARM_EABI_SYSCALL 0xef000000
|
|
|
|
static void
|
|
arm_linux_sigtramp_cache (struct frame_info *this_frame,
|
|
struct trad_frame_cache *this_cache,
|
|
CORE_ADDR func, int regs_offset)
|
|
{
|
|
CORE_ADDR sp = get_frame_register_unsigned (this_frame, ARM_SP_REGNUM);
|
|
CORE_ADDR base = sp + regs_offset;
|
|
int i;
|
|
|
|
for (i = 0; i < 16; i++)
|
|
trad_frame_set_reg_addr (this_cache, i, base + i * 4);
|
|
|
|
trad_frame_set_reg_addr (this_cache, ARM_PS_REGNUM, base + 16 * 4);
|
|
|
|
/* The VFP or iWMMXt registers may be saved on the stack, but there's
|
|
no reliable way to restore them (yet). */
|
|
|
|
/* Save a frame ID. */
|
|
trad_frame_set_id (this_cache, frame_id_build (sp, func));
|
|
}
|
|
|
|
/* There are a couple of different possible stack layouts that
|
|
we need to support.
|
|
|
|
Before version 2.6.18, the kernel used completely independent
|
|
layouts for non-RT and RT signals. For non-RT signals the stack
|
|
began directly with a struct sigcontext. For RT signals the stack
|
|
began with two redundant pointers (to the siginfo and ucontext),
|
|
and then the siginfo and ucontext.
|
|
|
|
As of version 2.6.18, the non-RT signal frame layout starts with
|
|
a ucontext and the RT signal frame starts with a siginfo and then
|
|
a ucontext. Also, the ucontext now has a designated save area
|
|
for coprocessor registers.
|
|
|
|
For RT signals, it's easy to tell the difference: we look for
|
|
pinfo, the pointer to the siginfo. If it has the expected
|
|
value, we have an old layout. If it doesn't, we have the new
|
|
layout.
|
|
|
|
For non-RT signals, it's a bit harder. We need something in one
|
|
layout or the other with a recognizable offset and value. We can't
|
|
use the return trampoline, because ARM usually uses SA_RESTORER,
|
|
in which case the stack return trampoline is not filled in.
|
|
We can't use the saved stack pointer, because sigaltstack might
|
|
be in use. So for now we guess the new layout... */
|
|
|
|
/* There are three words (trap_no, error_code, oldmask) in
|
|
struct sigcontext before r0. */
|
|
#define ARM_SIGCONTEXT_R0 0xc
|
|
|
|
/* There are five words (uc_flags, uc_link, and three for uc_stack)
|
|
in the ucontext_t before the sigcontext. */
|
|
#define ARM_UCONTEXT_SIGCONTEXT 0x14
|
|
|
|
/* There are three elements in an rt_sigframe before the ucontext:
|
|
pinfo, puc, and info. The first two are pointers and the third
|
|
is a struct siginfo, with size 128 bytes. We could follow puc
|
|
to the ucontext, but it's simpler to skip the whole thing. */
|
|
#define ARM_OLD_RT_SIGFRAME_SIGINFO 0x8
|
|
#define ARM_OLD_RT_SIGFRAME_UCONTEXT 0x88
|
|
|
|
#define ARM_NEW_RT_SIGFRAME_UCONTEXT 0x80
|
|
|
|
#define ARM_NEW_SIGFRAME_MAGIC 0x5ac3c35a
|
|
|
|
static void
|
|
arm_linux_sigreturn_init (const struct tramp_frame *self,
|
|
struct frame_info *this_frame,
|
|
struct trad_frame_cache *this_cache,
|
|
CORE_ADDR func)
|
|
{
|
|
struct gdbarch *gdbarch = get_frame_arch (this_frame);
|
|
enum bfd_endian byte_order = gdbarch_byte_order (gdbarch);
|
|
CORE_ADDR sp = get_frame_register_unsigned (this_frame, ARM_SP_REGNUM);
|
|
ULONGEST uc_flags = read_memory_unsigned_integer (sp, 4, byte_order);
|
|
|
|
if (uc_flags == ARM_NEW_SIGFRAME_MAGIC)
|
|
arm_linux_sigtramp_cache (this_frame, this_cache, func,
|
|
ARM_UCONTEXT_SIGCONTEXT
|
|
+ ARM_SIGCONTEXT_R0);
|
|
else
|
|
arm_linux_sigtramp_cache (this_frame, this_cache, func,
|
|
ARM_SIGCONTEXT_R0);
|
|
}
|
|
|
|
static void
|
|
arm_linux_rt_sigreturn_init (const struct tramp_frame *self,
|
|
struct frame_info *this_frame,
|
|
struct trad_frame_cache *this_cache,
|
|
CORE_ADDR func)
|
|
{
|
|
struct gdbarch *gdbarch = get_frame_arch (this_frame);
|
|
enum bfd_endian byte_order = gdbarch_byte_order (gdbarch);
|
|
CORE_ADDR sp = get_frame_register_unsigned (this_frame, ARM_SP_REGNUM);
|
|
ULONGEST pinfo = read_memory_unsigned_integer (sp, 4, byte_order);
|
|
|
|
if (pinfo == sp + ARM_OLD_RT_SIGFRAME_SIGINFO)
|
|
arm_linux_sigtramp_cache (this_frame, this_cache, func,
|
|
ARM_OLD_RT_SIGFRAME_UCONTEXT
|
|
+ ARM_UCONTEXT_SIGCONTEXT
|
|
+ ARM_SIGCONTEXT_R0);
|
|
else
|
|
arm_linux_sigtramp_cache (this_frame, this_cache, func,
|
|
ARM_NEW_RT_SIGFRAME_UCONTEXT
|
|
+ ARM_UCONTEXT_SIGCONTEXT
|
|
+ ARM_SIGCONTEXT_R0);
|
|
}
|
|
|
|
static struct tramp_frame arm_linux_sigreturn_tramp_frame = {
|
|
SIGTRAMP_FRAME,
|
|
4,
|
|
{
|
|
{ ARM_LINUX_SIGRETURN_INSTR, -1 },
|
|
{ TRAMP_SENTINEL_INSN }
|
|
},
|
|
arm_linux_sigreturn_init
|
|
};
|
|
|
|
static struct tramp_frame arm_linux_rt_sigreturn_tramp_frame = {
|
|
SIGTRAMP_FRAME,
|
|
4,
|
|
{
|
|
{ ARM_LINUX_RT_SIGRETURN_INSTR, -1 },
|
|
{ TRAMP_SENTINEL_INSN }
|
|
},
|
|
arm_linux_rt_sigreturn_init
|
|
};
|
|
|
|
static struct tramp_frame arm_eabi_linux_sigreturn_tramp_frame = {
|
|
SIGTRAMP_FRAME,
|
|
4,
|
|
{
|
|
{ ARM_SET_R7_SIGRETURN, -1 },
|
|
{ ARM_EABI_SYSCALL, -1 },
|
|
{ TRAMP_SENTINEL_INSN }
|
|
},
|
|
arm_linux_sigreturn_init
|
|
};
|
|
|
|
static struct tramp_frame arm_eabi_linux_rt_sigreturn_tramp_frame = {
|
|
SIGTRAMP_FRAME,
|
|
4,
|
|
{
|
|
{ ARM_SET_R7_RT_SIGRETURN, -1 },
|
|
{ ARM_EABI_SYSCALL, -1 },
|
|
{ TRAMP_SENTINEL_INSN }
|
|
},
|
|
arm_linux_rt_sigreturn_init
|
|
};
|
|
|
|
/* Core file and register set support. */
|
|
|
|
#define ARM_LINUX_SIZEOF_GREGSET (18 * INT_REGISTER_SIZE)
|
|
|
|
void
|
|
arm_linux_supply_gregset (const struct regset *regset,
|
|
struct regcache *regcache,
|
|
int regnum, const void *gregs_buf, size_t len)
|
|
{
|
|
struct gdbarch *gdbarch = get_regcache_arch (regcache);
|
|
enum bfd_endian byte_order = gdbarch_byte_order (gdbarch);
|
|
const gdb_byte *gregs = gregs_buf;
|
|
int regno;
|
|
CORE_ADDR reg_pc;
|
|
gdb_byte pc_buf[INT_REGISTER_SIZE];
|
|
|
|
for (regno = ARM_A1_REGNUM; regno < ARM_PC_REGNUM; regno++)
|
|
if (regnum == -1 || regnum == regno)
|
|
regcache_raw_supply (regcache, regno,
|
|
gregs + INT_REGISTER_SIZE * regno);
|
|
|
|
if (regnum == ARM_PS_REGNUM || regnum == -1)
|
|
{
|
|
if (arm_apcs_32)
|
|
regcache_raw_supply (regcache, ARM_PS_REGNUM,
|
|
gregs + INT_REGISTER_SIZE * ARM_CPSR_GREGNUM);
|
|
else
|
|
regcache_raw_supply (regcache, ARM_PS_REGNUM,
|
|
gregs + INT_REGISTER_SIZE * ARM_PC_REGNUM);
|
|
}
|
|
|
|
if (regnum == ARM_PC_REGNUM || regnum == -1)
|
|
{
|
|
reg_pc = extract_unsigned_integer (gregs
|
|
+ INT_REGISTER_SIZE * ARM_PC_REGNUM,
|
|
INT_REGISTER_SIZE, byte_order);
|
|
reg_pc = gdbarch_addr_bits_remove (gdbarch, reg_pc);
|
|
store_unsigned_integer (pc_buf, INT_REGISTER_SIZE, byte_order, reg_pc);
|
|
regcache_raw_supply (regcache, ARM_PC_REGNUM, pc_buf);
|
|
}
|
|
}
|
|
|
|
void
|
|
arm_linux_collect_gregset (const struct regset *regset,
|
|
const struct regcache *regcache,
|
|
int regnum, void *gregs_buf, size_t len)
|
|
{
|
|
gdb_byte *gregs = gregs_buf;
|
|
int regno;
|
|
|
|
for (regno = ARM_A1_REGNUM; regno < ARM_PC_REGNUM; regno++)
|
|
if (regnum == -1 || regnum == regno)
|
|
regcache_raw_collect (regcache, regno,
|
|
gregs + INT_REGISTER_SIZE * regno);
|
|
|
|
if (regnum == ARM_PS_REGNUM || regnum == -1)
|
|
{
|
|
if (arm_apcs_32)
|
|
regcache_raw_collect (regcache, ARM_PS_REGNUM,
|
|
gregs + INT_REGISTER_SIZE * ARM_CPSR_GREGNUM);
|
|
else
|
|
regcache_raw_collect (regcache, ARM_PS_REGNUM,
|
|
gregs + INT_REGISTER_SIZE * ARM_PC_REGNUM);
|
|
}
|
|
|
|
if (regnum == ARM_PC_REGNUM || regnum == -1)
|
|
regcache_raw_collect (regcache, ARM_PC_REGNUM,
|
|
gregs + INT_REGISTER_SIZE * ARM_PC_REGNUM);
|
|
}
|
|
|
|
/* Support for register format used by the NWFPE FPA emulator. */
|
|
|
|
#define typeNone 0x00
|
|
#define typeSingle 0x01
|
|
#define typeDouble 0x02
|
|
#define typeExtended 0x03
|
|
|
|
void
|
|
supply_nwfpe_register (struct regcache *regcache, int regno,
|
|
const gdb_byte *regs)
|
|
{
|
|
const gdb_byte *reg_data;
|
|
gdb_byte reg_tag;
|
|
gdb_byte buf[FP_REGISTER_SIZE];
|
|
|
|
reg_data = regs + (regno - ARM_F0_REGNUM) * FP_REGISTER_SIZE;
|
|
reg_tag = regs[(regno - ARM_F0_REGNUM) + NWFPE_TAGS_OFFSET];
|
|
memset (buf, 0, FP_REGISTER_SIZE);
|
|
|
|
switch (reg_tag)
|
|
{
|
|
case typeSingle:
|
|
memcpy (buf, reg_data, 4);
|
|
break;
|
|
case typeDouble:
|
|
memcpy (buf, reg_data + 4, 4);
|
|
memcpy (buf + 4, reg_data, 4);
|
|
break;
|
|
case typeExtended:
|
|
/* We want sign and exponent, then least significant bits,
|
|
then most significant. NWFPE does sign, most, least. */
|
|
memcpy (buf, reg_data, 4);
|
|
memcpy (buf + 4, reg_data + 8, 4);
|
|
memcpy (buf + 8, reg_data + 4, 4);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
regcache_raw_supply (regcache, regno, buf);
|
|
}
|
|
|
|
void
|
|
collect_nwfpe_register (const struct regcache *regcache, int regno,
|
|
gdb_byte *regs)
|
|
{
|
|
gdb_byte *reg_data;
|
|
gdb_byte reg_tag;
|
|
gdb_byte buf[FP_REGISTER_SIZE];
|
|
|
|
regcache_raw_collect (regcache, regno, buf);
|
|
|
|
/* NOTE drow/2006-06-07: This code uses the tag already in the
|
|
register buffer. I've preserved that when moving the code
|
|
from the native file to the target file. But this doesn't
|
|
always make sense. */
|
|
|
|
reg_data = regs + (regno - ARM_F0_REGNUM) * FP_REGISTER_SIZE;
|
|
reg_tag = regs[(regno - ARM_F0_REGNUM) + NWFPE_TAGS_OFFSET];
|
|
|
|
switch (reg_tag)
|
|
{
|
|
case typeSingle:
|
|
memcpy (reg_data, buf, 4);
|
|
break;
|
|
case typeDouble:
|
|
memcpy (reg_data, buf + 4, 4);
|
|
memcpy (reg_data + 4, buf, 4);
|
|
break;
|
|
case typeExtended:
|
|
memcpy (reg_data, buf, 4);
|
|
memcpy (reg_data + 4, buf + 8, 4);
|
|
memcpy (reg_data + 8, buf + 4, 4);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
void
|
|
arm_linux_supply_nwfpe (const struct regset *regset,
|
|
struct regcache *regcache,
|
|
int regnum, const void *regs_buf, size_t len)
|
|
{
|
|
const gdb_byte *regs = regs_buf;
|
|
int regno;
|
|
|
|
if (regnum == ARM_FPS_REGNUM || regnum == -1)
|
|
regcache_raw_supply (regcache, ARM_FPS_REGNUM,
|
|
regs + NWFPE_FPSR_OFFSET);
|
|
|
|
for (regno = ARM_F0_REGNUM; regno <= ARM_F7_REGNUM; regno++)
|
|
if (regnum == -1 || regnum == regno)
|
|
supply_nwfpe_register (regcache, regno, regs);
|
|
}
|
|
|
|
void
|
|
arm_linux_collect_nwfpe (const struct regset *regset,
|
|
const struct regcache *regcache,
|
|
int regnum, void *regs_buf, size_t len)
|
|
{
|
|
gdb_byte *regs = regs_buf;
|
|
int regno;
|
|
|
|
for (regno = ARM_F0_REGNUM; regno <= ARM_F7_REGNUM; regno++)
|
|
if (regnum == -1 || regnum == regno)
|
|
collect_nwfpe_register (regcache, regno, regs);
|
|
|
|
if (regnum == ARM_FPS_REGNUM || regnum == -1)
|
|
regcache_raw_collect (regcache, ARM_FPS_REGNUM,
|
|
regs + INT_REGISTER_SIZE * ARM_FPS_REGNUM);
|
|
}
|
|
|
|
/* Return the appropriate register set for the core section identified
|
|
by SECT_NAME and SECT_SIZE. */
|
|
|
|
static const struct regset *
|
|
arm_linux_regset_from_core_section (struct gdbarch *gdbarch,
|
|
const char *sect_name, size_t sect_size)
|
|
{
|
|
struct gdbarch_tdep *tdep = gdbarch_tdep (gdbarch);
|
|
|
|
if (strcmp (sect_name, ".reg") == 0
|
|
&& sect_size == ARM_LINUX_SIZEOF_GREGSET)
|
|
{
|
|
if (tdep->gregset == NULL)
|
|
tdep->gregset = regset_alloc (gdbarch, arm_linux_supply_gregset,
|
|
arm_linux_collect_gregset);
|
|
return tdep->gregset;
|
|
}
|
|
|
|
if (strcmp (sect_name, ".reg2") == 0
|
|
&& sect_size == ARM_LINUX_SIZEOF_NWFPE)
|
|
{
|
|
if (tdep->fpregset == NULL)
|
|
tdep->fpregset = regset_alloc (gdbarch, arm_linux_supply_nwfpe,
|
|
arm_linux_collect_nwfpe);
|
|
return tdep->fpregset;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/* Insert a single step breakpoint at the next executed instruction. */
|
|
|
|
static int
|
|
arm_linux_software_single_step (struct frame_info *frame)
|
|
{
|
|
struct gdbarch *gdbarch = get_frame_arch (frame);
|
|
struct address_space *aspace = get_frame_address_space (frame);
|
|
CORE_ADDR next_pc = arm_get_next_pc (frame, get_frame_pc (frame));
|
|
|
|
/* The Linux kernel offers some user-mode helpers in a high page. We can
|
|
not read this page (as of 2.6.23), and even if we could then we couldn't
|
|
set breakpoints in it, and even if we could then the atomic operations
|
|
would fail when interrupted. They are all called as functions and return
|
|
to the address in LR, so step to there instead. */
|
|
if (next_pc > 0xffff0000)
|
|
next_pc = get_frame_register_unsigned (frame, ARM_LR_REGNUM);
|
|
|
|
insert_single_step_breakpoint (gdbarch, aspace, next_pc);
|
|
|
|
return 1;
|
|
}
|
|
|
|
/* Support for displaced stepping of Linux SVC instructions. */
|
|
|
|
static void
|
|
arm_linux_cleanup_svc (struct gdbarch *gdbarch ATTRIBUTE_UNUSED,
|
|
struct regcache *regs,
|
|
struct displaced_step_closure *dsc)
|
|
{
|
|
CORE_ADDR from = dsc->insn_addr;
|
|
ULONGEST apparent_pc;
|
|
int within_scratch;
|
|
|
|
regcache_cooked_read_unsigned (regs, ARM_PC_REGNUM, &apparent_pc);
|
|
|
|
within_scratch = (apparent_pc >= dsc->scratch_base
|
|
&& apparent_pc < (dsc->scratch_base
|
|
+ DISPLACED_MODIFIED_INSNS * 4 + 4));
|
|
|
|
if (debug_displaced)
|
|
{
|
|
fprintf_unfiltered (gdb_stdlog, "displaced: PC is apparently %.8lx after "
|
|
"SVC step ", (unsigned long) apparent_pc);
|
|
if (within_scratch)
|
|
fprintf_unfiltered (gdb_stdlog, "(within scratch space)\n");
|
|
else
|
|
fprintf_unfiltered (gdb_stdlog, "(outside scratch space)\n");
|
|
}
|
|
|
|
if (within_scratch)
|
|
displaced_write_reg (regs, dsc, ARM_PC_REGNUM, from + 4, BRANCH_WRITE_PC);
|
|
}
|
|
|
|
static int
|
|
arm_linux_copy_svc (struct gdbarch *gdbarch, uint32_t insn, CORE_ADDR to,
|
|
struct regcache *regs, struct displaced_step_closure *dsc)
|
|
{
|
|
CORE_ADDR from = dsc->insn_addr;
|
|
struct frame_info *frame;
|
|
unsigned int svc_number = displaced_read_reg (regs, from, 7);
|
|
|
|
if (debug_displaced)
|
|
fprintf_unfiltered (gdb_stdlog, "displaced: copying Linux svc insn %.8lx\n",
|
|
(unsigned long) insn);
|
|
|
|
frame = get_current_frame ();
|
|
|
|
/* Is this a sigreturn or rt_sigreturn syscall? Note: these are only useful
|
|
for EABI. */
|
|
if (svc_number == 119 || svc_number == 173)
|
|
{
|
|
if (get_frame_type (frame) == SIGTRAMP_FRAME)
|
|
{
|
|
CORE_ADDR return_to;
|
|
struct symtab_and_line sal;
|
|
|
|
if (debug_displaced)
|
|
fprintf_unfiltered (gdb_stdlog, "displaced: found "
|
|
"sigreturn/rt_sigreturn SVC call. PC in frame = %lx\n",
|
|
(unsigned long) get_frame_pc (frame));
|
|
|
|
return_to = frame_unwind_caller_pc (frame);
|
|
if (debug_displaced)
|
|
fprintf_unfiltered (gdb_stdlog, "displaced: unwind pc = %lx. "
|
|
"Setting momentary breakpoint.\n", (unsigned long) return_to);
|
|
|
|
gdb_assert (inferior_thread ()->step_resume_breakpoint == NULL);
|
|
|
|
sal = find_pc_line (return_to, 0);
|
|
sal.pc = return_to;
|
|
sal.section = find_pc_overlay (return_to);
|
|
sal.explicit_pc = 1;
|
|
|
|
frame = get_prev_frame (frame);
|
|
|
|
if (frame)
|
|
{
|
|
inferior_thread ()->step_resume_breakpoint
|
|
= set_momentary_breakpoint (gdbarch, sal, get_frame_id (frame),
|
|
bp_step_resume);
|
|
|
|
/* We need to make sure we actually insert the momentary
|
|
breakpoint set above. */
|
|
insert_breakpoints ();
|
|
}
|
|
else if (debug_displaced)
|
|
fprintf_unfiltered (gdb_stderr, "displaced: couldn't find previous "
|
|
"frame to set momentary breakpoint for "
|
|
"sigreturn/rt_sigreturn\n");
|
|
}
|
|
else if (debug_displaced)
|
|
fprintf_unfiltered (gdb_stdlog, "displaced: sigreturn/rt_sigreturn "
|
|
"SVC call not in signal trampoline frame\n");
|
|
}
|
|
|
|
/* Preparation: If we detect sigreturn, set momentary breakpoint at resume
|
|
location, else nothing.
|
|
Insn: unmodified svc.
|
|
Cleanup: if pc lands in scratch space, pc <- insn_addr + 4
|
|
else leave pc alone. */
|
|
|
|
dsc->modinsn[0] = insn;
|
|
|
|
dsc->cleanup = &arm_linux_cleanup_svc;
|
|
/* Pretend we wrote to the PC, so cleanup doesn't set PC to the next
|
|
instruction. */
|
|
dsc->wrote_to_pc = 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* The following two functions implement single-stepping over calls to Linux
|
|
kernel helper routines, which perform e.g. atomic operations on architecture
|
|
variants which don't support them natively.
|
|
|
|
When this function is called, the PC will be pointing at the kernel helper
|
|
(at an address inaccessible to GDB), and r14 will point to the return
|
|
address. Displaced stepping always executes code in the copy area:
|
|
so, make the copy-area instruction branch back to the kernel helper (the
|
|
"from" address), and make r14 point to the breakpoint in the copy area. In
|
|
that way, we regain control once the kernel helper returns, and can clean
|
|
up appropriately (as if we had just returned from the kernel helper as it
|
|
would have been called from the non-displaced location). */
|
|
|
|
static void
|
|
cleanup_kernel_helper_return (struct gdbarch *gdbarch ATTRIBUTE_UNUSED,
|
|
struct regcache *regs,
|
|
struct displaced_step_closure *dsc)
|
|
{
|
|
displaced_write_reg (regs, dsc, ARM_LR_REGNUM, dsc->tmp[0], CANNOT_WRITE_PC);
|
|
displaced_write_reg (regs, dsc, ARM_PC_REGNUM, dsc->tmp[0], BRANCH_WRITE_PC);
|
|
}
|
|
|
|
static void
|
|
arm_catch_kernel_helper_return (struct gdbarch *gdbarch, CORE_ADDR from,
|
|
CORE_ADDR to, struct regcache *regs,
|
|
struct displaced_step_closure *dsc)
|
|
{
|
|
enum bfd_endian byte_order = gdbarch_byte_order (gdbarch);
|
|
|
|
dsc->numinsns = 1;
|
|
dsc->insn_addr = from;
|
|
dsc->cleanup = &cleanup_kernel_helper_return;
|
|
/* Say we wrote to the PC, else cleanup will set PC to the next
|
|
instruction in the helper, which isn't helpful. */
|
|
dsc->wrote_to_pc = 1;
|
|
|
|
/* Preparation: tmp[0] <- r14
|
|
r14 <- <scratch space>+4
|
|
*(<scratch space>+8) <- from
|
|
Insn: ldr pc, [r14, #4]
|
|
Cleanup: r14 <- tmp[0], pc <- tmp[0]. */
|
|
|
|
dsc->tmp[0] = displaced_read_reg (regs, from, ARM_LR_REGNUM);
|
|
displaced_write_reg (regs, dsc, ARM_LR_REGNUM, (ULONGEST) to + 4,
|
|
CANNOT_WRITE_PC);
|
|
write_memory_unsigned_integer (to + 8, 4, byte_order, from);
|
|
|
|
dsc->modinsn[0] = 0xe59ef004; /* ldr pc, [lr, #4]. */
|
|
}
|
|
|
|
/* Linux-specific displaced step instruction copying function. Detects when
|
|
the program has stepped into a Linux kernel helper routine (which must be
|
|
handled as a special case), falling back to arm_displaced_step_copy_insn()
|
|
if it hasn't. */
|
|
|
|
static struct displaced_step_closure *
|
|
arm_linux_displaced_step_copy_insn (struct gdbarch *gdbarch,
|
|
CORE_ADDR from, CORE_ADDR to,
|
|
struct regcache *regs)
|
|
{
|
|
struct displaced_step_closure *dsc
|
|
= xmalloc (sizeof (struct displaced_step_closure));
|
|
|
|
/* Detect when we enter an (inaccessible by GDB) Linux kernel helper, and
|
|
stop at the return location. */
|
|
if (from > 0xffff0000)
|
|
{
|
|
if (debug_displaced)
|
|
fprintf_unfiltered (gdb_stdlog, "displaced: detected kernel helper "
|
|
"at %.8lx\n", (unsigned long) from);
|
|
|
|
arm_catch_kernel_helper_return (gdbarch, from, to, regs, dsc);
|
|
}
|
|
else
|
|
{
|
|
enum bfd_endian byte_order = gdbarch_byte_order (gdbarch);
|
|
uint32_t insn = read_memory_unsigned_integer (from, 4, byte_order);
|
|
|
|
if (debug_displaced)
|
|
fprintf_unfiltered (gdb_stdlog, "displaced: stepping insn %.8lx "
|
|
"at %.8lx\n", (unsigned long) insn,
|
|
(unsigned long) from);
|
|
|
|
/* Override the default handling of SVC instructions. */
|
|
dsc->u.svc.copy_svc_os = arm_linux_copy_svc;
|
|
|
|
arm_process_displaced_insn (gdbarch, insn, from, to, regs, dsc);
|
|
}
|
|
|
|
arm_displaced_init_closure (gdbarch, from, to, dsc);
|
|
|
|
return dsc;
|
|
}
|
|
|
|
static void
|
|
arm_linux_init_abi (struct gdbarch_info info,
|
|
struct gdbarch *gdbarch)
|
|
{
|
|
struct gdbarch_tdep *tdep = gdbarch_tdep (gdbarch);
|
|
|
|
tdep->lowest_pc = 0x8000;
|
|
if (info.byte_order == BFD_ENDIAN_BIG)
|
|
{
|
|
if (tdep->arm_abi == ARM_ABI_AAPCS)
|
|
tdep->arm_breakpoint = eabi_linux_arm_be_breakpoint;
|
|
else
|
|
tdep->arm_breakpoint = arm_linux_arm_be_breakpoint;
|
|
tdep->thumb_breakpoint = arm_linux_thumb_be_breakpoint;
|
|
}
|
|
else
|
|
{
|
|
if (tdep->arm_abi == ARM_ABI_AAPCS)
|
|
tdep->arm_breakpoint = eabi_linux_arm_le_breakpoint;
|
|
else
|
|
tdep->arm_breakpoint = arm_linux_arm_le_breakpoint;
|
|
tdep->thumb_breakpoint = arm_linux_thumb_le_breakpoint;
|
|
}
|
|
tdep->arm_breakpoint_size = sizeof (arm_linux_arm_le_breakpoint);
|
|
tdep->thumb_breakpoint_size = sizeof (arm_linux_thumb_le_breakpoint);
|
|
|
|
if (tdep->fp_model == ARM_FLOAT_AUTO)
|
|
tdep->fp_model = ARM_FLOAT_FPA;
|
|
|
|
tdep->jb_pc = ARM_LINUX_JB_PC;
|
|
tdep->jb_elt_size = ARM_LINUX_JB_ELEMENT_SIZE;
|
|
|
|
set_solib_svr4_fetch_link_map_offsets
|
|
(gdbarch, svr4_ilp32_fetch_link_map_offsets);
|
|
|
|
/* Single stepping. */
|
|
set_gdbarch_software_single_step (gdbarch, arm_linux_software_single_step);
|
|
|
|
/* Shared library handling. */
|
|
set_gdbarch_skip_trampoline_code (gdbarch, find_solib_trampoline_target);
|
|
set_gdbarch_skip_solib_resolver (gdbarch, glibc_skip_solib_resolver);
|
|
|
|
/* Enable TLS support. */
|
|
set_gdbarch_fetch_tls_load_module_address (gdbarch,
|
|
svr4_fetch_objfile_link_map);
|
|
|
|
tramp_frame_prepend_unwinder (gdbarch,
|
|
&arm_linux_sigreturn_tramp_frame);
|
|
tramp_frame_prepend_unwinder (gdbarch,
|
|
&arm_linux_rt_sigreturn_tramp_frame);
|
|
tramp_frame_prepend_unwinder (gdbarch,
|
|
&arm_eabi_linux_sigreturn_tramp_frame);
|
|
tramp_frame_prepend_unwinder (gdbarch,
|
|
&arm_eabi_linux_rt_sigreturn_tramp_frame);
|
|
|
|
/* Core file support. */
|
|
set_gdbarch_regset_from_core_section (gdbarch,
|
|
arm_linux_regset_from_core_section);
|
|
|
|
set_gdbarch_get_siginfo_type (gdbarch, linux_get_siginfo_type);
|
|
|
|
/* Displaced stepping. */
|
|
set_gdbarch_displaced_step_copy_insn (gdbarch,
|
|
arm_linux_displaced_step_copy_insn);
|
|
set_gdbarch_displaced_step_fixup (gdbarch, arm_displaced_step_fixup);
|
|
set_gdbarch_displaced_step_free_closure (gdbarch,
|
|
simple_displaced_step_free_closure);
|
|
set_gdbarch_displaced_step_location (gdbarch, displaced_step_at_entry_point);
|
|
}
|
|
|
|
/* Provide a prototype to silence -Wmissing-prototypes. */
|
|
extern initialize_file_ftype _initialize_arm_linux_tdep;
|
|
|
|
void
|
|
_initialize_arm_linux_tdep (void)
|
|
{
|
|
gdbarch_register_osabi (bfd_arch_arm, 0, GDB_OSABI_LINUX,
|
|
arm_linux_init_abi);
|
|
}
|