forked from Imagelibrary/binutils-gdb
This patch makes GDBserver support multi-process + biarch. Currently, if you're debugging more than one process at once with a single gdbserver (in extended-remote mode), then all processes must have the same architecture (e.g., 64-bit vs 32-bit). Otherwise, you see this: Added inferior 2 [Switching to inferior 2 [<null>] (<noexec>)] Reading symbols from /home/pedro/gdb/tests/main32...done. Temporary breakpoint 2 at 0x4004cf: main. (2 locations) Starting program: /home/pedro/gdb/tests/main32 warning: Selected architecture i386 is not compatible with reported target architecture i386:x86-64 warning: Architecture rejected target-supplied description Remote 'g' packet reply is too long: 000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000090cfffff0000000000000000000000000000000000000000000000000000000000020000000000000000000000000000000000000000000000000000000000000000000000000000b042f7460000000000020000230000002b0000002b0000002b000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000007f03000000000000ffff0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000801f00003b0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 ... etc, etc ... Even though the process was running a 32-bit program, GDBserver sent back to GDB a register set in 64-bit layout. A patch (http://sourceware.org/ml/gdb-patches/2012-11/msg00228.html) a while ago made GDB track a target_gdbarch per inferior, and as consequence, fetch a target description per-inferior. This patch is the GDBserver counterpart, that makes GDBserver keep track of each process'es XML target description and register layout. So in the example above, GDBserver will send the correct register set in 32-bit layout to GDB. A new "struct target_desc" object (tdesc for short) is added, that holds the target description and register layout information about each process. Each `struct process_info' holds a pointer to a target description. The regcache also gains a pointer to a target description, mainly for convenience, and parallel with GDB (and possible future support for programs that flip processor modes). The low target's arch_setup routines are responsible for setting the process'es correct tdesc. This isn't that much different to how things were done before, except that instead of detecting the inferior process'es architecture and calling the corresponding init_registers_FOO routine, which would change the regcache layout globals and recreate the threads' regcaches, the regcache.c globals are gone, and the init_registers_$BAR routines now each initialize a separate global struct target_desc object (one for each arch variant GDBserver supports), and so all the init_registers_$BAR routines that are built into GDBserver are called early at GDBserver startup time (similarly to how GDB handles its built-in target descriptions), and then the arch_setup routine is responsible for making process_info->tdesc point to one of these target description globals. The regcache module is all parameterized to get the regcache's layout from the tdesc object instead of the old register_bytes, etc. globals. The threads' regcaches are now created lazily. The old scheme where we created each of them when we added a new thread doesn't work anymore, because we add the main thread/lwp before we see it stop for the first time, and it is only when we see the thread stop for the first time that we have a chance of determining the inferior's architecture (through the_low_target.arch_setup). Therefore when we add the main thread we don't know which architecture/tdesc its regcache should have. This patch makes the gdb.multi/multi-arch.exp test now pass against (extended-remote) GDBserver. It currently fails, without this patch. The IPA also uses the regcache, so it gains a new global struct target_desc pointer, which points at the description of the process it is loaded in. Re. the linux-low.c & friends changes. Since the register map etc. may differ between processes (64-bit vs 32-bit) etc., the linux_target_ops num_regs, regmap and regset_bitmap data fields are no longer sufficient. A new method is added in their place that returns a pointer to a new struct that includes all info linux-low.c needs to access registers of the current inferior. The patch/discussion that originally introduced linux-low.c:disabled_regsets mentions that the disabled_regsets set may be different per mode (in a biarch setup), and indeed that is cleared whenever we start a new (first) inferior, so that global is moved as well behind the new `struct regs_info'. On the x86 side: I simply replaced the i387-fp.c:num_xmm_registers global with a check for 64-bit or 32-bit process, which is equivalent to how the global was set. This avoided coming up with some more general mechanism that would work for all targets that use this module (GNU/Linux, Windows, etc.). Tested: GNU/Linux IA64 GNU/Linux MIPS64 GNU/Linux PowerPC (Fedora 16) GNU/Linux s390x (Fedora 16) GNU/Linux sparc64 (Debian) GNU/Linux x86_64, -m64 and -m32 (Fedora 17) Cross built, and smoke tested: i686-w64-mingw32, under Wine. GNU/Linux TI C6x, by Yao Qi. Cross built but otherwise not tested: aarch64-linux-gnu arm-linux-gnu m68k-linux nios2-linux-gnu sh-linux-gnu spu tilegx-unknown-linux-gnu Completely untested: GNU/Linux Blackfin GNU/Linux CRIS GNU/Linux CRISv32 GNU/Linux TI Xtensa GNU/Linux M32R LynxOS QNX NTO gdb/gdbserver/ 2013-06-07 Pedro Alves <palves@redhat.com> * Makefile.in (OBS): Add tdesc.o. (IPA_OBJS): Add tdesc-ipa.o. (tdesc-ipa.o): New rule. * ax.c (gdb_eval_agent_expr): Adjust register_size call to new interface. * linux-low.c (new_inferior): Delete. (disabled_regsets, num_regsets): Delete. (linux_add_process): Adjust to set the new per-process new_inferior flag. (linux_detach_one_lwp): Adjust to call regcache_invalidate_thread. (linux_wait_for_lwp): Adjust. Only call arch_setup if the event was a stop. When calling arch_setup, switch the current inferior to the thread that got an event. (linux_resume_one_lwp): Adjust to call regcache_invalidate_thread. (regsets_fetch_inferior_registers) (regsets_store_inferior_registers): New regsets_info parameter. Adjust to use it. (linux_register_in_regsets): New regs_info parameter. Adjust to use it. (register_addr, fetch_register, store_register): New usrregs_info parameter. Adjust to use it. (usr_fetch_inferior_registers, usr_store_inferior_registers): New parameter regs_info. Adjust to use it. (linux_fetch_registers): Get the current inferior's regs_info, and adjust to use it. (linux_store_registers): Ditto. [HAVE_LINUX_REGSETS] (initialize_regsets_info): New. (initialize_low): Don't initialize the target_regsets here. Call initialize_low_arch. * linux-low.h (target_regsets): Delete declaration. (struct regsets_info): New. (struct usrregs_info): New. (struct regs_info): New. (struct process_info_private) <new_inferior>: New field. (struct linux_target_ops): Delete the num_regs, regmap, and regset_bitmap fields. New field regs_info. [HAVE_LINUX_REGSETS] (initialize_regsets_info): Declare. * i387-fp.c (num_xmm_registers): Delete. (i387_cache_to_fsave, i387_fsave_to_cache): Adjust find_regno calls to new interface. (i387_cache_to_fxsave, i387_cache_to_xsave, i387_fxsave_to_cache) (i387_xsave_to_cache): Adjust find_regno calls to new interface. Infer the number of xmm registers from the regcache's target description. * i387-fp.h (num_xmm_registers): Delete. * inferiors.c (add_thread): Don't install the thread's regcache here. * proc-service.c (gregset_info): Fetch the current inferior's regs_info. Adjust to use it. * regcache.c: Include tdesc.h. (register_bytes, reg_defs, num_registers) (gdbserver_expedite_regs): Delete. (get_thread_regcache): If the thread doesn't have a regcache yet, create one, instead of aborting gdbserver. (regcache_invalidate_one): Rename to ... (regcache_invalidate_thread): ... this. (regcache_invalidate_one): New. (regcache_invalidate): Only invalidate registers of the current process. (init_register_cache): Add target_desc parameter, and use it. (new_register_cache): Ditto. Assert the target description has a non zero registers_size. (regcache_cpy): Add assertions. Adjust. (realloc_register_cache, set_register_cache): Delete. (registers_to_string, registers_from_string): Adjust. (find_register_by_name, find_regno, find_register_by_number) (register_cache_size): Add target_desc parameter, and use it. (free_register_cache_thread, free_register_cache_thread_one) (regcache_release, register_cache_size): New. (register_size): Add target_desc parameter, and use it. (register_data, supply_register, supply_register_zeroed) (supply_regblock, supply_register_by_name, collect_register) (collect_register_as_string, collect_register_by_name): Adjust. * regcache.h (struct target_desc): Forward declare. (struct regcache) <tdesc>: New field. (init_register_cache, new_register_cache): Add target_desc parameter. (regcache_invalidate_thread): Declare. (regcache_invalidate_one): Delete declaration. (regcache_release): Declare. (find_register_by_number, register_cache_size, register_size) (find_regno): Add target_desc parameter. (gdbserver_expedite_regs, gdbserver_xmltarget): Delete declarations. * remote-utils.c: Include tdesc.h. (outreg, prepare_resume_reply): Adjust. * server.c: Include tdesc.h. (gdbserver_xmltarget): Delete declaration. (get_features_xml, process_serial_event): Adjust. * server.h [IN_PROCESS_AGENT] (struct target_desc): Forward declare. (struct process_info) <tdesc>: New field. (ipa_tdesc): Declare. * tdesc.c: New file. * tdesc.h: New file. * tracepoint.c: Include tdesc.h. [IN_PROCESS_AGENT] (ipa_tdesc): Define. (get_context_regcache): Adjust to pass ipa_tdesc down. (do_action_at_tracepoint): Adjust to get the register cache size from the context regcache's description. (traceframe_walk_blocks): Adjust to get the register cache size from the current trace frame's description. (traceframe_get_pc): Adjust to get current trace frame's description and pass it down. (gdb_collect): Adjust to get the register cache size from the IPA's description. * linux-amd64-ipa.c (tdesc_amd64_linux): Declare. (gdbserver_xmltarget): Delete. (initialize_low_tracepoint): Set the ipa's target description. * linux-i386-ipa.c (tdesc_i386_linux): Declare. (initialize_low_tracepoint): Set the ipa's target description. * linux-x86-low.c: Include tdesc.h. [__x86_64__] (is_64bit_tdesc): New. (ps_get_thread_area, x86_get_thread_area): Use it. (i386_cannot_store_register): Rename to ... (x86_cannot_store_register): ... this. Use is_64bit_tdesc. (i386_cannot_fetch_register): Rename to ... (x86_cannot_fetch_register): ... this. Use is_64bit_tdesc. (x86_fill_gregset, x86_store_gregset): Adjust register_size calls to new interface. (target_regsets): Rename to ... (x86_regsets): ... this. (x86_get_pc, x86_set_pc): Adjust register_size calls to new interface. (x86_siginfo_fixup): Use is_64bit_tdesc. [__x86_64__] (tdesc_amd64_linux, tdesc_amd64_avx_linux) (tdesc_x32_avx_linux, tdesc_x32_linux) (tdesc_i386_linux, tdesc_i386_mmx_linux, tdesc_i386_avx_linux): Declare. (x86_linux_update_xmltarget): Delete. (I386_LINUX_XSAVE_XCR0_OFFSET): Define. (have_ptrace_getfpxregs, have_ptrace_getregset): New. (AMD64_LINUX_USER64_CS): New. (x86_linux_read_description): New, based on x86_linux_update_xmltarget. (same_process_callback): New. (x86_arch_setup_process_callback): New. (x86_linux_update_xmltarget): New. (x86_regsets_info): New. (amd64_linux_regs_info): New. (i386_linux_usrregs_info): New. (i386_linux_regs_info): New. (x86_linux_regs_info): New. (x86_arch_setup): Reimplement. (x86_install_fast_tracepoint_jump_pad): Use is_64bit_tdesc. (x86_emit_ops): Ditto. (the_low_target): Adjust. Install x86_linux_regs_info, x86_cannot_fetch_register, and x86_cannot_store_register. (initialize_low_arch): New. * linux-ia64-low.c (tdesc_ia64): Declare. (ia64_fetch_register): Adjust. (ia64_usrregs_info, regs_info): New globals. (ia64_regs_info): New function. (the_low_target): Adjust. (initialize_low_arch): New function. * linux-sparc-low.c (tdesc_sparc64): Declare. (sparc_fill_gregset_to_stack, sparc_store_gregset_from_stack): Adjust. (sparc_arch_setup): New function. (sparc_regsets_info, sparc_usrregs_info, regs_info): New globals. (the_low_target): Adjust. (initialize_low_arch): New function. * linux-ppc-low.c (tdesc_powerpc_32l, tdesc_powerpc_altivec32l) (tdesc_powerpc_cell32l, tdesc_powerpc_vsx32l) (tdesc_powerpc_isa205_32l, tdesc_powerpc_isa205_altivec32l) (tdesc_powerpc_isa205_vsx32l, tdesc_powerpc_e500l) (tdesc_powerpc_64l, tdesc_powerpc_altivec64l) (tdesc_powerpc_cell64l, tdesc_powerpc_vsx64l) (tdesc_powerpc_isa205_64l, tdesc_powerpc_isa205_altivec64l) (tdesc_powerpc_isa205_vsx64l): Declare. (ppc_cannot_store_register, ppc_collect_ptrace_register) (ppc_supply_ptrace_register, parse_spufs_run, ppc_get_pc) (ppc_set_pc, ppc_get_hwcap): Adjust. (ppc_usrregs_info): Forward declare. (!__powerpc64__) ppc_regmap_adjusted: New global. (ppc_arch_setup): Adjust to the current process'es target description. (ppc_fill_vsxregset, ppc_store_vsxregset, ppc_fill_vrregset) (ppc_store_vrregset, ppc_fill_evrregset, ppc_store_evrregse) (ppc_store_evrregset): Adjust. (target_regsets): Rename to ... (ppc_regsets): ... this, and make static. (ppc_usrregs_info, ppc_regsets_info, regs_info): New globals. (ppc_regs_info): New function. (the_low_target): Adjust. (initialize_low_arch): New function. * linux-s390-low.c (tdesc_s390_linux32, tdesc_s390_linux32v1) (tdesc_s390_linux32v2, tdesc_s390_linux64, tdesc_s390_linux64v1) (tdesc_s390_linux64v2, tdesc_s390x_linux64, tdesc_s390x_linux64v1) (tdesc_s390x_linux64v2): Declare. (s390_collect_ptrace_register, s390_supply_ptrace_register) (s390_fill_gregset, s390_store_last_break): Adjust. (target_regsets): Rename to ... (s390_regsets): ... this, and make static. (s390_get_pc, s390_set_pc): Adjust. (s390_get_hwcap): New target_desc parameter, and use it. [__s390x__] (have_hwcap_s390_high_gprs): New global. (s390_arch_setup): Adjust to set the current process'es target description. Don't adjust the regmap. (s390_usrregs_info, s390_regsets_info, regs_info): New globals. [__s390x__] (s390_usrregs_info_3264, s390_regsets_info_3264) (regs_info_3264): New globals. (s390_regs_info): New function. (the_low_target): Adjust. (initialize_low_arch): New function. * linux-mips-low.c (tdesc_mips_linux, tdesc_mips_dsp_linux) (tdesc_mips64_linux, tdesc_mips64_dsp_linux): Declare. [__mips64] (init_registers_mips_linux) (init_registers_mips_dsp_linux): Delete defines. [__mips64] (tdesc_mips_linux, tdesc_mips_dsp_linux): New defines. (have_dsp): New global. (mips_read_description): New, based on mips_arch_setup. (mips_arch_setup): Reimplement. (get_usrregs_info): New function. (mips_cannot_fetch_register, mips_cannot_store_register) (mips_get_pc, mips_set_pc, mips_fill_gregset, mips_store_gregset) (mips_fill_fpregset, mips_store_fpregset): Adjust. (target_regsets): Rename to ... (mips_regsets): ... this, and make static. (mips_regsets_info, mips_dsp_usrregs_info, mips_usrregs_info) (dsp_regs_info, regs_info): New globals. (mips_regs_info): New function. (the_low_target): Adjust. (initialize_low_arch): New function. * linux-arm-low.c (tdesc_arm, tdesc_arm_with_iwmmxt) (tdesc_arm_with_vfpv2, tdesc_arm_with_vfpv3, tdesc_arm_with_neon): Declare. (arm_fill_vfpregset, arm_store_vfpregset): Adjust. (arm_read_description): New, with bits factored from arm_arch_setup. (arm_arch_setup): Reimplement. (target_regsets): Rename to ... (arm_regsets): ... this, and make static. (arm_regsets_info, arm_usrregs_info, regs_info): New globals. (arm_regs_info): New function. (the_low_target): Adjust. (initialize_low_arch): New function. * linux-m68k-low.c (tdesc_m68k): Declare. (target_regsets): Rename to ... (m68k_regsets): ... this, and make static. (m68k_regsets_info, m68k_usrregs_info, regs_info): New globals. (m68k_regs_info): New function. (m68k_arch_setup): New function. (the_low_target): Adjust. (initialize_low_arch): New function. * linux-sh-low.c (tdesc_sharch): Declare. (target_regsets): Rename to ... (sh_regsets): ... this, and make static. (sh_regsets_info, sh_usrregs_info, regs_info): New globals. (sh_regs_info, sh_arch_setup): New functions. (the_low_target): Adjust. (initialize_low_arch): New function. * linux-bfin-low.c (tdesc_bfin): Declare. (bfin_arch_setup): New function. (bfin_usrregs_info, regs_info): New globals. (bfin_regs_info): New function. (the_low_target): Adjust. (initialize_low_arch): New function. * linux-cris-low.c (tdesc_cris): Declare. (cris_arch_setup): New function. (cris_usrregs_info, regs_info): New globals. (cris_regs_info): New function. (the_low_target): Adjust. (initialize_low_arch): New function. * linux-cris-low.c (tdesc_crisv32): Declare. (cris_arch_setup): New function. (cris_regsets_info, cris_usrregs_info, regs_info): New globals. (cris_regs_info): New function. (the_low_target): Adjust. (initialize_low_arch): New function. * linux-m32r-low.c (tdesc_m32r): Declare. (m32r_arch_setup): New function. (m32r_usrregs_info, regs_info): New globals. (m32r_regs_info): Adjust. (initialize_low_arch): New function. * linux-tic6x-low.c (tdesc_tic6x_c64xp_linux) (tdesc_tic6x_c64x_linux, tdesc_tic6x_c62x_linux): Declare. (tic6x_usrregs_info): Forward declare. (tic6x_read_description): New function, based on ... (tic6x_arch_setup): ... this. Reimplement. (target_regsets): Rename to ... (tic6x_regsets): ... this, and make static. (tic6x_regsets_info, tic6x_usrregs_info, regs_info): New globals. (tic6x_regs_info): New function. (the_low_target): Adjust. (initialize_low_arch): New function. * linux-xtensa-low.c (tdesc_xtensa): Declare. (xtensa_fill_gregset, xtensa_store_gregset): Adjust. (target_regsets): Rename to ... (xtensa_regsets): ... this, and make static. (xtensa_regsets_info, xtensa_usrregs_info, regs_info): New globals. (xtensa_arch_setup, xtensa_regs_info): New functions. (the_low_target): Adjust. (initialize_low_arch): New function. * linux-nios2-low.c (tdesc_nios2_linux): Declare. (nios2_arch_setup): Set the current process'es tdesc. (target_regsets): Rename to ... (nios2_regsets): ... this. (nios2_regsets_info, nios2_usrregs_info, regs_info): New globals. (nios2_regs_info): New function. (the_low_target): Adjust. (initialize_low_arch): New function. * linux-aarch64-low.c (tdesc_aarch64): Declare. (aarch64_arch_setup): Set the current process'es tdesc. (target_regsets): Rename to ... (aarch64_regsets): ... this. (aarch64_regsets_info, aarch64_usrregs_info, regs_info): New globals. (aarch64_regs_info): New function. (the_low_target): Adjust. (initialize_low_arch): New function. * linux-tile-low.c (tdesc_tilegx, tdesc_tilegx32): Declare globals. (target_regsets): Rename to ... (tile_regsets): ... this. (tile_regsets_info, tile_usrregs_info, regs_info): New globals. (tile_regs_info): New function. (tile_arch_setup): Set the current process'es tdesc. (the_low_target): Adjust. (initialize_low_arch): New function. * spu-low.c (tdesc_spu): Declare. (spu_create_inferior, spu_attach): Set the new process'es tdesc. * win32-arm-low.c (tdesc_arm): Declare. (arm_arch_setup): New function. (the_low_target): Install arm_arch_setup instead of init_registers_arm. * win32-i386-low.c (tdesc_i386, tdesc_amd64): Declare. (init_windows_x86): Rename to ... (i386_arch_setup): ... this. Set `win32_tdesc'. (the_low_target): Adjust. * win32-low.c (win32_tdesc): New global. (child_add_thread): Don't create the thread cache here. (do_initial_child_stuff): Set the new process'es tdesc. * win32-low.h (struct target_desc): Forward declare. (win32_tdesc): Declare. * lynx-i386-low.c (tdesc_i386): Declare global. (lynx_i386_arch_setup): Set `lynx_tdesc'. * lynx-low.c (lynx_tdesc): New global. (lynx_add_process): Set the new process'es tdesc. * lynx-low.h (struct target_desc): Forward declare. (lynx_tdesc): Declare global. * lynx-ppc-low.c (tdesc_powerpc_32): Declare global. (lynx_ppc_arch_setup): Set `lynx_tdesc'. * nto-low.c (nto_tdesc): New global. (do_attach): Set the new process'es tdesc. * nto-low.h (struct target_desc): Forward declare. (nto_tdesc): Declare. * nto-x86-low.c (tdesc_i386): Declare. (nto_x86_arch_setup): Set `nto_tdesc'. gdb/ 2013-06-07 Pedro Alves <palves@redhat.com> * regformats/regdat.sh: Output #include tdesc.h. Make globals static. Output a global target description pointer. (init_registers_${name}): Adjust to initialize a target description structure.
539 lines
16 KiB
C
539 lines
16 KiB
C
/* Common definitions for remote server for GDB.
|
|
Copyright (C) 1993-2013 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/>. */
|
|
|
|
#ifndef SERVER_H
|
|
#define SERVER_H
|
|
|
|
#include "config.h"
|
|
#include "build-gnulib-gdbserver/config.h"
|
|
|
|
#ifdef __MINGW32CE__
|
|
#include "wincecompat.h"
|
|
#endif
|
|
|
|
#include "libiberty.h"
|
|
#include "ansidecl.h"
|
|
|
|
#include <stdarg.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#ifdef HAVE_ERRNO_H
|
|
#include <errno.h>
|
|
#endif
|
|
#include <setjmp.h>
|
|
|
|
#ifdef HAVE_STRING_H
|
|
#include <string.h>
|
|
#endif
|
|
|
|
#ifdef HAVE_ALLOCA_H
|
|
#include <alloca.h>
|
|
#endif
|
|
/* On some systems such as MinGW, alloca is declared in malloc.h
|
|
(there is no alloca.h). */
|
|
#if HAVE_MALLOC_H
|
|
#include <malloc.h>
|
|
#endif
|
|
|
|
#if !HAVE_DECL_STRERROR
|
|
#ifndef strerror
|
|
extern char *strerror (int); /* X3.159-1989 4.11.6.2 */
|
|
#endif
|
|
#endif
|
|
|
|
#if !HAVE_DECL_PERROR
|
|
#ifndef perror
|
|
extern void perror (const char *);
|
|
#endif
|
|
#endif
|
|
|
|
#if !HAVE_DECL_VASPRINTF
|
|
extern int vasprintf(char **strp, const char *fmt, va_list ap);
|
|
#endif
|
|
#if !HAVE_DECL_VSNPRINTF
|
|
int vsnprintf(char *str, size_t size, const char *format, va_list ap);
|
|
#endif
|
|
|
|
/* Define underscore macro, if not available, to be able to use it inside
|
|
code shared with gdb in common directory. */
|
|
#ifndef _
|
|
#define _(String) (String)
|
|
#endif
|
|
|
|
#ifdef IN_PROCESS_AGENT
|
|
# define PROG "ipa"
|
|
#else
|
|
# define PROG "gdbserver"
|
|
#endif
|
|
|
|
/* A type used for binary buffers. */
|
|
typedef unsigned char gdb_byte;
|
|
|
|
#include "ptid.h"
|
|
#include "buffer.h"
|
|
#include "xml-utils.h"
|
|
#include "gdb_locale.h"
|
|
|
|
/* FIXME: This should probably be autoconf'd for. It's an integer type at
|
|
least the size of a (void *). */
|
|
typedef long long CORE_ADDR;
|
|
|
|
typedef long long LONGEST;
|
|
typedef unsigned long long ULONGEST;
|
|
|
|
/* Generic information for tracking a list of ``inferiors'' - threads,
|
|
processes, etc. */
|
|
struct inferior_list
|
|
{
|
|
struct inferior_list_entry *head;
|
|
struct inferior_list_entry *tail;
|
|
};
|
|
struct inferior_list_entry
|
|
{
|
|
ptid_t id;
|
|
struct inferior_list_entry *next;
|
|
};
|
|
|
|
struct thread_info;
|
|
struct process_info;
|
|
struct regcache;
|
|
struct target_desc;
|
|
|
|
#include "regcache.h"
|
|
#include "gdb/signals.h"
|
|
#include "gdb_signals.h"
|
|
#include "target.h"
|
|
#include "mem-break.h"
|
|
#include "gdbthread.h"
|
|
|
|
struct dll_info
|
|
{
|
|
struct inferior_list_entry entry;
|
|
char *name;
|
|
CORE_ADDR base_addr;
|
|
};
|
|
|
|
struct sym_cache;
|
|
struct breakpoint;
|
|
struct raw_breakpoint;
|
|
struct fast_tracepoint_jump;
|
|
struct process_info_private;
|
|
|
|
struct process_info
|
|
{
|
|
struct inferior_list_entry head;
|
|
|
|
/* Nonzero if this child process was attached rather than
|
|
spawned. */
|
|
int attached;
|
|
|
|
/* True if GDB asked us to detach from this process, but we remained
|
|
attached anyway. */
|
|
int gdb_detached;
|
|
|
|
/* The symbol cache. */
|
|
struct sym_cache *symbol_cache;
|
|
|
|
/* The list of memory breakpoints. */
|
|
struct breakpoint *breakpoints;
|
|
|
|
/* The list of raw memory breakpoints. */
|
|
struct raw_breakpoint *raw_breakpoints;
|
|
|
|
/* The list of installed fast tracepoints. */
|
|
struct fast_tracepoint_jump *fast_tracepoint_jumps;
|
|
|
|
const struct target_desc *tdesc;
|
|
|
|
/* Private target data. */
|
|
struct process_info_private *private;
|
|
};
|
|
|
|
/* Return a pointer to the process that corresponds to the current
|
|
thread (current_inferior). It is an error to call this if there is
|
|
no current thread selected. */
|
|
|
|
struct process_info *current_process (void);
|
|
struct process_info *get_thread_process (struct thread_info *);
|
|
|
|
/* Target-specific functions */
|
|
|
|
void initialize_low ();
|
|
|
|
/* From inferiors.c. */
|
|
|
|
extern struct inferior_list all_processes;
|
|
extern struct inferior_list all_dlls;
|
|
extern int dlls_changed;
|
|
extern void clear_dlls (void);
|
|
|
|
void add_inferior_to_list (struct inferior_list *list,
|
|
struct inferior_list_entry *new_inferior);
|
|
void for_each_inferior (struct inferior_list *list,
|
|
void (*action) (struct inferior_list_entry *));
|
|
|
|
extern struct thread_info *current_inferior;
|
|
void remove_inferior (struct inferior_list *list,
|
|
struct inferior_list_entry *entry);
|
|
|
|
struct process_info *add_process (int pid, int attached);
|
|
void remove_process (struct process_info *process);
|
|
struct process_info *find_process_pid (int pid);
|
|
int have_started_inferiors_p (void);
|
|
int have_attached_inferiors_p (void);
|
|
|
|
ptid_t thread_id_to_gdb_id (ptid_t);
|
|
ptid_t thread_to_gdb_id (struct thread_info *);
|
|
ptid_t gdb_id_to_thread_id (ptid_t);
|
|
|
|
void clear_inferiors (void);
|
|
struct inferior_list_entry *find_inferior
|
|
(struct inferior_list *,
|
|
int (*func) (struct inferior_list_entry *,
|
|
void *),
|
|
void *arg);
|
|
struct inferior_list_entry *find_inferior_id (struct inferior_list *list,
|
|
ptid_t id);
|
|
void *inferior_target_data (struct thread_info *);
|
|
void set_inferior_target_data (struct thread_info *, void *);
|
|
void *inferior_regcache_data (struct thread_info *);
|
|
void set_inferior_regcache_data (struct thread_info *, void *);
|
|
|
|
void loaded_dll (const char *name, CORE_ADDR base_addr);
|
|
void unloaded_dll (const char *name, CORE_ADDR base_addr);
|
|
|
|
/* Public variables in server.c */
|
|
|
|
extern ptid_t cont_thread;
|
|
extern ptid_t general_thread;
|
|
|
|
extern int server_waiting;
|
|
extern int debug_threads;
|
|
extern int debug_hw_points;
|
|
extern int pass_signals[];
|
|
extern int program_signals[];
|
|
extern int program_signals_p;
|
|
|
|
extern jmp_buf toplevel;
|
|
|
|
extern int disable_packet_vCont;
|
|
extern int disable_packet_Tthread;
|
|
extern int disable_packet_qC;
|
|
extern int disable_packet_qfThreadInfo;
|
|
|
|
extern int run_once;
|
|
extern int multi_process;
|
|
extern int non_stop;
|
|
|
|
extern int disable_randomization;
|
|
|
|
#if USE_WIN32API
|
|
#include <winsock2.h>
|
|
typedef SOCKET gdb_fildes_t;
|
|
#else
|
|
typedef int gdb_fildes_t;
|
|
#endif
|
|
|
|
/* Functions from event-loop.c. */
|
|
typedef void *gdb_client_data;
|
|
typedef int (handler_func) (int, gdb_client_data);
|
|
typedef int (callback_handler_func) (gdb_client_data);
|
|
|
|
extern void delete_file_handler (gdb_fildes_t fd);
|
|
extern void add_file_handler (gdb_fildes_t fd, handler_func *proc,
|
|
gdb_client_data client_data);
|
|
extern int append_callback_event (callback_handler_func *proc,
|
|
gdb_client_data client_data);
|
|
extern void delete_callback_event (int id);
|
|
|
|
extern void start_event_loop (void);
|
|
extern void initialize_event_loop (void);
|
|
|
|
/* Functions from server.c. */
|
|
extern int handle_serial_event (int err, gdb_client_data client_data);
|
|
extern int handle_target_event (int err, gdb_client_data client_data);
|
|
|
|
/* Functions from hostio.c. */
|
|
extern int handle_vFile (char *, int, int *);
|
|
|
|
/* Functions from hostio-errno.c. */
|
|
extern void hostio_last_error_from_errno (char *own_buf);
|
|
|
|
/* From remote-utils.c */
|
|
|
|
extern int remote_debug;
|
|
extern int noack_mode;
|
|
extern int transport_is_reliable;
|
|
|
|
int gdb_connected (void);
|
|
|
|
#define STDIO_CONNECTION_NAME "stdio"
|
|
int remote_connection_is_stdio (void);
|
|
|
|
ptid_t read_ptid (char *buf, char **obuf);
|
|
char *write_ptid (char *buf, ptid_t ptid);
|
|
|
|
int putpkt (char *buf);
|
|
int putpkt_binary (char *buf, int len);
|
|
int putpkt_notif (char *buf);
|
|
int getpkt (char *buf);
|
|
void remote_prepare (char *name);
|
|
void remote_open (char *name);
|
|
void remote_close (void);
|
|
void write_ok (char *buf);
|
|
void write_enn (char *buf);
|
|
void initialize_async_io (void);
|
|
void enable_async_io (void);
|
|
void disable_async_io (void);
|
|
void check_remote_input_interrupt_request (void);
|
|
void convert_ascii_to_int (const char *from, unsigned char *to, int n);
|
|
void convert_int_to_ascii (const unsigned char *from, char *to, int n);
|
|
void new_thread_notify (int id);
|
|
void dead_thread_notify (int id);
|
|
void prepare_resume_reply (char *buf, ptid_t ptid,
|
|
struct target_waitstatus *status);
|
|
|
|
const char *decode_address_to_semicolon (CORE_ADDR *addrp, const char *start);
|
|
void decode_address (CORE_ADDR *addrp, const char *start, int len);
|
|
void decode_m_packet (char *from, CORE_ADDR * mem_addr_ptr,
|
|
unsigned int *len_ptr);
|
|
void decode_M_packet (char *from, CORE_ADDR * mem_addr_ptr,
|
|
unsigned int *len_ptr, unsigned char **to_p);
|
|
int decode_X_packet (char *from, int packet_len, CORE_ADDR * mem_addr_ptr,
|
|
unsigned int *len_ptr, unsigned char **to_p);
|
|
int decode_xfer_write (char *buf, int packet_len,
|
|
CORE_ADDR *offset, unsigned int *len,
|
|
unsigned char *data);
|
|
int decode_search_memory_packet (const char *buf, int packet_len,
|
|
CORE_ADDR *start_addrp,
|
|
CORE_ADDR *search_space_lenp,
|
|
gdb_byte *pattern,
|
|
unsigned int *pattern_lenp);
|
|
|
|
int unhexify (char *bin, const char *hex, int count);
|
|
int hexify (char *hex, const char *bin, int count);
|
|
int remote_escape_output (const gdb_byte *buffer, int len,
|
|
gdb_byte *out_buf, int *out_len,
|
|
int out_maxlen);
|
|
char *unpack_varlen_hex (char *buff, ULONGEST *result);
|
|
|
|
void clear_symbol_cache (struct sym_cache **symcache_p);
|
|
int look_up_one_symbol (const char *name, CORE_ADDR *addrp, int may_ask_gdb);
|
|
|
|
int relocate_instruction (CORE_ADDR *to, CORE_ADDR oldloc);
|
|
|
|
void monitor_output (const char *msg);
|
|
|
|
/* Functions from utils.c */
|
|
#include "common-utils.h"
|
|
|
|
void perror_with_name (const char *string);
|
|
void error (const char *string,...) ATTRIBUTE_NORETURN ATTRIBUTE_PRINTF (1, 2);
|
|
void fatal (const char *string,...) ATTRIBUTE_NORETURN ATTRIBUTE_PRINTF (1, 2);
|
|
void warning (const char *string,...) ATTRIBUTE_PRINTF (1, 2);
|
|
char *paddress (CORE_ADDR addr);
|
|
char *pulongest (ULONGEST u);
|
|
char *plongest (LONGEST l);
|
|
char *phex_nz (ULONGEST l, int sizeof_l);
|
|
char *pfildes (gdb_fildes_t fd);
|
|
|
|
#include "gdb_assert.h"
|
|
|
|
/* Maximum number of bytes to read/write at once. The value here
|
|
is chosen to fill up a packet (the headers account for the 32). */
|
|
#define MAXBUFBYTES(N) (((N)-32)/2)
|
|
|
|
/* Buffer sizes for transferring memory, registers, etc. Set to a constant
|
|
value to accomodate multiple register formats. This value must be at least
|
|
as large as the largest register set supported by gdbserver. */
|
|
#define PBUFSIZ 16384
|
|
|
|
/* Functions from tracepoint.c */
|
|
|
|
/* Size for a small buffer to report problems from the in-process
|
|
agent back to GDBserver. */
|
|
#define IPA_BUFSIZ 100
|
|
|
|
void initialize_tracepoint (void);
|
|
|
|
extern int tracing;
|
|
extern int disconnected_tracing;
|
|
|
|
void tracepoint_look_up_symbols (void);
|
|
|
|
void stop_tracing (void);
|
|
|
|
int handle_tracepoint_general_set (char *own_buf);
|
|
int handle_tracepoint_query (char *own_buf);
|
|
|
|
int tracepoint_finished_step (struct thread_info *tinfo, CORE_ADDR stop_pc);
|
|
int tracepoint_was_hit (struct thread_info *tinfo, CORE_ADDR stop_pc);
|
|
|
|
void release_while_stepping_state_list (struct thread_info *tinfo);
|
|
|
|
extern int current_traceframe;
|
|
|
|
int in_readonly_region (CORE_ADDR addr, ULONGEST length);
|
|
int traceframe_read_mem (int tfnum, CORE_ADDR addr,
|
|
unsigned char *buf, ULONGEST length,
|
|
ULONGEST *nbytes);
|
|
int fetch_traceframe_registers (int tfnum,
|
|
struct regcache *regcache,
|
|
int regnum);
|
|
|
|
int traceframe_read_sdata (int tfnum, ULONGEST offset,
|
|
unsigned char *buf, ULONGEST length,
|
|
ULONGEST *nbytes);
|
|
|
|
int traceframe_read_info (int tfnum, struct buffer *buffer);
|
|
|
|
/* If a thread is determined to be collecting a fast tracepoint, this
|
|
structure holds the collect status. */
|
|
|
|
struct fast_tpoint_collect_status
|
|
{
|
|
/* The tracepoint that is presently being collected. */
|
|
int tpoint_num;
|
|
CORE_ADDR tpoint_addr;
|
|
|
|
/* The address range in the jump pad of where the original
|
|
instruction the tracepoint jump was inserted was relocated
|
|
to. */
|
|
CORE_ADDR adjusted_insn_addr;
|
|
CORE_ADDR adjusted_insn_addr_end;
|
|
};
|
|
|
|
int fast_tracepoint_collecting (CORE_ADDR thread_area,
|
|
CORE_ADDR stop_pc,
|
|
struct fast_tpoint_collect_status *status);
|
|
void force_unlock_trace_buffer (void);
|
|
|
|
int handle_tracepoint_bkpts (struct thread_info *tinfo, CORE_ADDR stop_pc);
|
|
|
|
#ifdef IN_PROCESS_AGENT
|
|
void initialize_low_tracepoint (void);
|
|
void supply_fast_tracepoint_registers (struct regcache *regcache,
|
|
const unsigned char *regs);
|
|
void supply_static_tracepoint_registers (struct regcache *regcache,
|
|
const unsigned char *regs,
|
|
CORE_ADDR pc);
|
|
void set_trampoline_buffer_space (CORE_ADDR begin, CORE_ADDR end,
|
|
char *errmsg);
|
|
|
|
extern const struct target_desc *ipa_tdesc;
|
|
|
|
#else
|
|
void stop_tracing (void);
|
|
|
|
int claim_trampoline_space (ULONGEST used, CORE_ADDR *trampoline);
|
|
int have_fast_tracepoint_trampoline_buffer (char *msgbuf);
|
|
void gdb_agent_about_to_close (int pid);
|
|
#endif
|
|
|
|
struct traceframe;
|
|
struct eval_agent_expr_context;
|
|
|
|
/* Do memory copies for bytecodes. */
|
|
/* Do the recording of memory blocks for actions and bytecodes. */
|
|
|
|
int agent_mem_read (struct eval_agent_expr_context *ctx,
|
|
unsigned char *to, CORE_ADDR from,
|
|
ULONGEST len);
|
|
|
|
LONGEST agent_get_trace_state_variable_value (int num);
|
|
void agent_set_trace_state_variable_value (int num, LONGEST val);
|
|
|
|
/* Record the value of a trace state variable. */
|
|
|
|
int agent_tsv_read (struct eval_agent_expr_context *ctx, int n);
|
|
int agent_mem_read_string (struct eval_agent_expr_context *ctx,
|
|
unsigned char *to,
|
|
CORE_ADDR from,
|
|
ULONGEST len);
|
|
|
|
/* Bytecode compilation function vector. */
|
|
|
|
struct emit_ops
|
|
{
|
|
void (*emit_prologue) (void);
|
|
void (*emit_epilogue) (void);
|
|
void (*emit_add) (void);
|
|
void (*emit_sub) (void);
|
|
void (*emit_mul) (void);
|
|
void (*emit_lsh) (void);
|
|
void (*emit_rsh_signed) (void);
|
|
void (*emit_rsh_unsigned) (void);
|
|
void (*emit_ext) (int arg);
|
|
void (*emit_log_not) (void);
|
|
void (*emit_bit_and) (void);
|
|
void (*emit_bit_or) (void);
|
|
void (*emit_bit_xor) (void);
|
|
void (*emit_bit_not) (void);
|
|
void (*emit_equal) (void);
|
|
void (*emit_less_signed) (void);
|
|
void (*emit_less_unsigned) (void);
|
|
void (*emit_ref) (int size);
|
|
void (*emit_if_goto) (int *offset_p, int *size_p);
|
|
void (*emit_goto) (int *offset_p, int *size_p);
|
|
void (*write_goto_address) (CORE_ADDR from, CORE_ADDR to, int size);
|
|
void (*emit_const) (LONGEST num);
|
|
void (*emit_call) (CORE_ADDR fn);
|
|
void (*emit_reg) (int reg);
|
|
void (*emit_pop) (void);
|
|
void (*emit_stack_flush) (void);
|
|
void (*emit_zero_ext) (int arg);
|
|
void (*emit_swap) (void);
|
|
void (*emit_stack_adjust) (int n);
|
|
|
|
/* Emit code for a generic function that takes one fixed integer
|
|
argument and returns a 64-bit int (for instance, tsv getter). */
|
|
void (*emit_int_call_1) (CORE_ADDR fn, int arg1);
|
|
|
|
/* Emit code for a generic function that takes one fixed integer
|
|
argument and a 64-bit int from the top of the stack, and returns
|
|
nothing (for instance, tsv setter). */
|
|
void (*emit_void_call_2) (CORE_ADDR fn, int arg1);
|
|
|
|
/* Emit code specialized for common combinations of compare followed
|
|
by a goto. */
|
|
void (*emit_eq_goto) (int *offset_p, int *size_p);
|
|
void (*emit_ne_goto) (int *offset_p, int *size_p);
|
|
void (*emit_lt_goto) (int *offset_p, int *size_p);
|
|
void (*emit_le_goto) (int *offset_p, int *size_p);
|
|
void (*emit_gt_goto) (int *offset_p, int *size_p);
|
|
void (*emit_ge_goto) (int *offset_p, int *size_p);
|
|
};
|
|
|
|
/* Returns the address of the get_raw_reg function in the IPA. */
|
|
CORE_ADDR get_raw_reg_func_addr (void);
|
|
/* Returns the address of the get_trace_state_variable_value
|
|
function in the IPA. */
|
|
CORE_ADDR get_get_tsv_func_addr (void);
|
|
/* Returns the address of the set_trace_state_variable_value
|
|
function in the IPA. */
|
|
CORE_ADDR get_set_tsv_func_addr (void);
|
|
|
|
extern CORE_ADDR current_insn_ptr;
|
|
extern int emit_error;
|
|
|
|
/* Version information, from version.c. */
|
|
extern const char version[];
|
|
extern const char host_name[];
|
|
|
|
#endif /* SERVER_H */
|