8d7493201c
This patch is part of the make-gdb-buildable-in-C++ effort. The idea is to change some calls to the xmalloc family of functions to calls to the equivalents in the XNEW family. This avoids adding an explicit cast, so it keeps the code a bit more readable. Some of them also map relatively well to a C++ equivalent (XNEW (struct foo) -> new foo), so it will be possible to do scripted replacements if needed. I only changed calls that were obviously allocating memory for one or multiple "objects". Allocation of variable sizes (such as strings or buffer handling) will be for later (and won't use XNEW). - xmalloc (sizeof (struct foo)) -> XNEW (struct foo) - xmalloc (num * sizeof (struct foo)) -> XNEWVEC (struct foo, num) - xcalloc (1, sizeof (struct foo)) -> XCNEW (struct foo) - xcalloc (num, sizeof (struct foo)) -> XCNEWVEC (struct foo, num) - xrealloc (p, num * sizeof (struct foo) -> XRESIZEVEC (struct foo, p, num) - obstack_alloc (ob, sizeof (struct foo)) -> XOBNEW (ob, struct foo) - obstack_alloc (ob, num * sizeof (struct foo)) -> XOBNEWVEC (ob, struct foo, num) - alloca (sizeof (struct foo)) -> XALLOCA (struct foo) - alloca (num * sizeof (struct foo)) -> XALLOCAVEC (struct foo, num) Some instances of xmalloc followed by memset to zero the buffer were replaced by XCNEW or XCNEWVEC. I regtested on x86-64, Ubuntu 14.04, but the patch touches many architecture-specific files. For those I'll have to rely on the buildbot or people complaining that I broke their gdb. gdb/ChangeLog: * aarch64-linux-nat.c (aarch64_add_process): Likewise. * aarch64-tdep.c (aarch64_gdbarch_init): Likewise. * ada-exp.y (write_ambiguous_var): Likewise. * ada-lang.c (resolve_subexp): Likewise. (user_select_syms): Likewise. (assign_aggregate): Likewise. (ada_evaluate_subexp): Likewise. (cache_symbol): Likewise. * addrmap.c (allocate_key): Likewise. (addrmap_create_mutable): Likewise. * aix-thread.c (sync_threadlists): Likewise. * alpha-tdep.c (alpha_push_dummy_call): Likewise. (alpha_gdbarch_init): Likewise. * amd64-windows-tdep.c (amd64_windows_push_arguments): Likewise. * arm-linux-nat.c (arm_linux_add_process): Likewise. * arm-linux-tdep.c (arm_linux_displaced_step_copy_insn): Likewise. * arm-tdep.c (push_stack_item): Likewise. (arm_displaced_step_copy_insn): Likewise. (arm_gdbarch_init): Likewise. (_initialize_arm_tdep): Likewise. * avr-tdep.c (push_stack_item): Likewise. * ax-general.c (new_agent_expr): Likewise. * block.c (block_initialize_namespace): Likewise. * breakpoint.c (alloc_counted_command_line): Likewise. (update_dprintf_command_list): Likewise. (parse_breakpoint_sals): Likewise. (decode_static_tracepoint_spec): Likewise. (until_break_command): Likewise. (clear_command): Likewise. (update_global_location_list): Likewise. (get_breakpoint_objfile_data) Likewise. * btrace.c (ftrace_new_function): Likewise. (btrace_set_insn_history): Likewise. (btrace_set_call_history): Likewise. * buildsym.c (add_symbol_to_list): Likewise. (record_pending_block): Likewise. (start_subfile): Likewise. (start_buildsym_compunit): Likewise. (push_subfile): Likewise. (end_symtab_get_static_block): Likewise. (buildsym_init): Likewise. * cli/cli-cmds.c (source_command): Likewise. * cli/cli-decode.c (add_cmd): Likewise. * cli/cli-script.c (build_command_line): Likewise. (setup_user_args): Likewise. (realloc_body_list): Likewise. (process_next_line): Likewise. (copy_command_lines): Likewise. * cli/cli-setshow.c (do_set_command): Likewise. * coff-pe-read.c (read_pe_exported_syms): Likewise. * coffread.c (coff_locate_sections): Likewise. (coff_symtab_read): Likewise. (coff_read_struct_type): Likewise. * common/cleanups.c (make_my_cleanup2): Likewise. * common/common-exceptions.c (throw_it): Likewise. * common/filestuff.c (make_cleanup_close): Likewise. * common/format.c (parse_format_string): Likewise. * common/queue.h (DEFINE_QUEUE_P): Likewise. * compile/compile-object-load.c (munmap_list_add): Likewise. (compile_object_load): Likewise. * compile/compile-object-run.c (compile_object_run): Likewise. * compile/compile.c (append_args): Likewise. * corefile.c (specify_exec_file_hook): Likewise. * cp-support.c (make_symbol_overload_list): Likewise. * cris-tdep.c (push_stack_item): Likewise. (cris_gdbarch_init): Likewise. * ctf.c (ctf_trace_file_writer_new): Likewise. * dbxread.c (init_header_files): Likewise. (add_new_header_file): Likewise. (init_bincl_list): Likewise. (dbx_end_psymtab): Likewise. (start_psymtab): Likewise. (dbx_end_psymtab): Likewise. * dcache.c (dcache_init): Likewise. * dictionary.c (dict_create_hashed): Likewise. (dict_create_hashed_expandable): Likewise. (dict_create_linear): Likewise. (dict_create_linear_expandable): Likewise. * dtrace-probe.c (dtrace_process_dof_probe): Likewise. * dummy-frame.c (register_dummy_frame_dtor): Likewise. * dwarf2-frame-tailcall.c (cache_new_ref1): Likewise. * dwarf2-frame.c (dwarf2_build_frame_info): Likewise. (decode_frame_entry_1): Likewise. * dwarf2expr.c (new_dwarf_expr_context): Likewise. * dwarf2loc.c (dwarf2_compile_expr_to_ax): Likewise. * dwarf2read.c (dwarf2_has_info): Likewise. (create_signatured_type_table_from_index): Likewise. (dwarf2_read_index): Likewise. (dw2_get_file_names_reader): Likewise. (create_all_type_units): Likewise. (read_cutu_die_from_dwo): Likewise. (init_tu_and_read_dwo_dies): Likewise. (init_cutu_and_read_dies): Likewise. (create_all_comp_units): Likewise. (queue_comp_unit): Likewise. (inherit_abstract_dies): Likewise. (read_call_site_scope): Likewise. (dwarf2_add_field): Likewise. (dwarf2_add_typedef): Likewise. (dwarf2_add_member_fn): Likewise. (attr_to_dynamic_prop): Likewise. (abbrev_table_alloc_abbrev): Likewise. (abbrev_table_read_table): Likewise. (add_include_dir): Likewise. (add_file_name): Likewise. (dwarf_decode_line_header): Likewise. (dwarf2_const_value_attr): Likewise. (dwarf_alloc_block): Likewise. (parse_macro_definition): Likewise. (set_die_type): Likewise. (write_psymtabs_to_index): Likewise. (create_cus_from_index): Likewise. (dwarf2_create_include_psymtab): Likewise. (process_psymtab_comp_unit_reader): Likewise. (build_type_psymtab_dependencies): Likewise. (read_comp_units_from_section): Likewise. (compute_compunit_symtab_includes): Likewise. (create_dwo_unit_in_dwp_v1): Likewise. (create_dwo_unit_in_dwp_v2): Likewise. (read_func_scope): Likewise. (process_structure_scope): Likewise. (mark_common_block_symbol_computed): Likewise. (load_partial_dies): Likewise. (dwarf2_symbol_mark_computed): Likewise. * elfread.c (elf_symfile_segments): Likewise. (elf_read_minimal_symbols): Likewise. * environ.c (make_environ): Likewise. * eval.c (evaluate_subexp_standard): Likewise. * event-loop.c (create_file_handler): Likewise. (create_async_signal_handler): Likewise. (create_async_event_handler): Likewise. (create_timer): Likewise. * exec.c (build_section_table): Likewise. * fbsd-nat.c (fbsd_remember_child): Likewise. * fork-child.c (fork_inferior): Likewise. * frv-tdep.c (new_variant): Likewise. * gdbarch.sh (gdbarch_alloc): Likewise. (append_name): Likewise. * gdbtypes.c (rank_function): Likewise. (copy_type_recursive): Likewise. (add_dyn_prop): Likewise. * gnu-nat.c (make_proc): Likewise. (make_inf): Likewise. (gnu_write_inferior): Likewise. * gnu-v3-abi.c (build_gdb_vtable_type): Likewise. (build_std_type_info_type): Likewise. * guile/scm-param.c (compute_enum_list): Likewise. * guile/scm-utils.c (gdbscm_parse_function_args): Likewise. * guile/scm-value.c (gdbscm_value_call): Likewise. * h8300-tdep.c (h8300_gdbarch_init): Likewise. * hppa-tdep.c (hppa_init_objfile_priv_data): Likewise. (read_unwind_info): Likewise. * ia64-tdep.c (ia64_gdbarch_init): Likewise. * infcall.c (dummy_frame_context_saver_setup): Likewise. (call_function_by_hand_dummy): Likewise. * infcmd.c (step_once): Likewise. (finish_forward): Likewise. (attach_command): Likewise. (notice_new_inferior): Likewise. * inferior.c (add_inferior_silent): Likewise. * infrun.c (add_displaced_stepping_state): Likewise. (save_infcall_control_state): Likewise. (save_inferior_ptid): Likewise. (_initialize_infrun): Likewise. * jit.c (bfd_open_from_target_memory): Likewise. (jit_gdbarch_data_init): Likewise. * language.c (add_language): Likewise. * linespec.c (decode_line_2): Likewise. * linux-nat.c (add_to_pid_list): Likewise. (add_initial_lwp): Likewise. * linux-thread-db.c (add_thread_db_info): Likewise. (record_thread): Likewise. (info_auto_load_libthread_db): Likewise. * m32c-tdep.c (m32c_gdbarch_init): Likewise. * m68hc11-tdep.c (m68hc11_gdbarch_init): Likewise. * m68k-tdep.c (m68k_gdbarch_init): Likewise. * m88k-tdep.c (m88k_analyze_prologue): Likewise. * macrocmd.c (macro_define_command): Likewise. * macroexp.c (gather_arguments): Likewise. * macroscope.c (sal_macro_scope): Likewise. * macrotab.c (new_macro_table): Likewise. * mdebugread.c (push_parse_stack): Likewise. (parse_partial_symbols): Likewise. (parse_symbol): Likewise. (psymtab_to_symtab_1): Likewise. (new_block): Likewise. (new_psymtab): Likewise. (mdebug_build_psymtabs): Likewise. (add_pending): Likewise. (elfmdebug_build_psymtabs): Likewise. * mep-tdep.c (mep_gdbarch_init): Likewise. * mi/mi-main.c (mi_execute_command): Likewise. * mi/mi-parse.c (mi_parse_argv): Likewise. * minidebug.c (lzma_open): Likewise. * minsyms.c (terminate_minimal_symbol_table): Likewise. * mips-linux-nat.c (mips_linux_insert_watchpoint): Likewise. * mips-tdep.c (mips_gdbarch_init): Likewise. * mn10300-tdep.c (mn10300_gdbarch_init): Likewise. * msp430-tdep.c (msp430_gdbarch_init): Likewise. * mt-tdep.c (mt_registers_info): Likewise. * nat/aarch64-linux.c (aarch64_linux_new_thread): Likewise. * nat/linux-btrace.c (linux_enable_bts): Likewise. (linux_enable_pt): Likewise. * nat/linux-osdata.c (linux_xfer_osdata_processes): Likewise. (linux_xfer_osdata_processgroups): Likewise. * nios2-tdep.c (nios2_gdbarch_init): Likewise. * nto-procfs.c (procfs_meminfo): Likewise. * objc-lang.c (start_msglist): Likewise. (selectors_info): Likewise. (classes_info): Likewise. (find_methods): Likewise. * objfiles.c (allocate_objfile): Likewise. (update_section_map): Likewise. * osabi.c (gdbarch_register_osabi): Likewise. (gdbarch_register_osabi_sniffer): Likewise. * parse.c (start_arglist): Likewise. * ppc-linux-nat.c (hwdebug_find_thread_points_by_tid): Likewise. (hwdebug_insert_point): Likewise. * printcmd.c (display_command): Likewise. (ui_printf): Likewise. * procfs.c (create_procinfo): Likewise. (load_syscalls): Likewise. (proc_get_LDT_entry): Likewise. (proc_update_threads): Likewise. * prologue-value.c (make_pv_area): Likewise. (pv_area_store): Likewise. * psymtab.c (extend_psymbol_list): Likewise. (init_psymbol_list): Likewise. (allocate_psymtab): Likewise. * python/py-inferior.c (add_thread_object): Likewise. * python/py-param.c (compute_enum_values): Likewise. * python/py-value.c (valpy_call): Likewise. * python/py-varobj.c (py_varobj_iter_next): Likewise. * python/python.c (ensure_python_env): Likewise. * record-btrace.c (record_btrace_start_replaying): Likewise. * record-full.c (record_full_reg_alloc): Likewise. (record_full_mem_alloc): Likewise. (record_full_end_alloc): Likewise. (record_full_core_xfer_partial): Likewise. * regcache.c (get_thread_arch_aspace_regcache): Likewise. * remote-fileio.c (remote_fileio_init_fd_map): Likewise. * remote-notif.c (remote_notif_state_allocate): Likewise. * remote.c (demand_private_info): Likewise. (remote_notif_stop_alloc_reply): Likewise. (remote_enable_btrace): Likewise. * reverse.c (save_bookmark_command): Likewise. * rl78-tdep.c (rl78_gdbarch_init): Likewise. * rx-tdep.c (rx_gdbarch_init): Likewise. * s390-linux-nat.c (s390_insert_watchpoint): Likewise. * ser-go32.c (dos_get_tty_state): Likewise. (dos_copy_tty_state): Likewise. * ser-mingw.c (ser_windows_open): Likewise. (ser_console_wait_handle): Likewise. (ser_console_get_tty_state): Likewise. (make_pipe_state): Likewise. (net_windows_open): Likewise. * ser-unix.c (hardwire_get_tty_state): Likewise. (hardwire_copy_tty_state): Likewise. * solib-aix.c (solib_aix_new_lm_info): Likewise. * solib-dsbt.c (dsbt_current_sos): Likewise. (dsbt_relocate_main_executable): Likewise. * solib-frv.c (frv_current_sos): Likewise. (frv_relocate_main_executable): Likewise. * solib-spu.c (spu_bfd_fopen): Likewise. * solib-svr4.c (lm_info_read): Likewise. (svr4_copy_library_list): Likewise. (svr4_default_sos): Likewise. * source.c (find_source_lines): Likewise. (line_info): Likewise. (add_substitute_path_rule): Likewise. * spu-linux-nat.c (spu_bfd_open): Likewise. * spu-tdep.c (info_spu_dma_cmdlist): Likewise. * stabsread.c (dbx_lookup_type): Likewise. (read_type): Likewise. (read_member_functions): Likewise. (read_struct_fields): Likewise. (read_baseclasses): Likewise. (read_args): Likewise. (_initialize_stabsread): Likewise. * stack.c (func_command): Likewise. * stap-probe.c (handle_stap_probe): Likewise. * symfile.c (addrs_section_sort): Likewise. (addr_info_make_relative): Likewise. (load_section_callback): Likewise. (add_symbol_file_command): Likewise. (init_filename_language_table): Likewise. * symtab.c (create_filename_seen_cache): Likewise. (sort_search_symbols_remove_dups): Likewise. (search_symbols): Likewise. * target.c (make_cleanup_restore_target_terminal): Likewise. * thread.c (new_thread): Likewise. (enable_thread_stack_temporaries): Likewise. (make_cleanup_restore_current_thread): Likewise. (thread_apply_all_command): Likewise. * tic6x-tdep.c (tic6x_gdbarch_init): Likewise. * top.c (gdb_readline_wrapper): Likewise. * tracefile-tfile.c (tfile_trace_file_writer_new): Likewise. * tracepoint.c (trace_find_line_command): Likewise. (all_tracepoint_actions_and_cleanup): Likewise. (make_cleanup_restore_current_traceframe): Likewise. (get_uploaded_tp): Likewise. (get_uploaded_tsv): Likewise. * tui/tui-data.c (tui_alloc_generic_win_info): Likewise. (tui_alloc_win_info): Likewise. (tui_alloc_content): Likewise. (tui_add_content_elements): Likewise. * tui/tui-disasm.c (tui_find_disassembly_address): Likewise. (tui_set_disassem_content): Likewise. * ui-file.c (ui_file_new): Likewise. (stdio_file_new): Likewise. (tee_file_new): Likewise. * utils.c (make_cleanup_restore_integer): Likewise. (add_internal_problem_command): Likewise. * v850-tdep.c (v850_gdbarch_init): Likewise. * valops.c (find_oload_champ): Likewise. * value.c (allocate_value_lazy): Likewise. (record_latest_value): Likewise. (create_internalvar): Likewise. * varobj.c (install_variable): Likewise. (new_variable): Likewise. (new_root_variable): Likewise. (cppush): Likewise. (_initialize_varobj): Likewise. * windows-nat.c (windows_make_so): Likewise. * x86-nat.c (x86_add_process): Likewise. * xcoffread.c (arrange_linetable): Likewise. (allocate_include_entry): Likewise. (process_linenos): Likewise. (SYMBOL_DUP): Likewise. (xcoff_start_psymtab): Likewise. (xcoff_end_psymtab): Likewise. * xml-support.c (gdb_xml_parse_attr_ulongest): Likewise. * xtensa-tdep.c (xtensa_register_type): Likewise. * gdbarch.c: Regenerate. * gdbarch.h: Regenerate. gdb/gdbserver/ChangeLog: * ax.c (gdb_parse_agent_expr): Likewise. (compile_bytecodes): Likewise. * dll.c (loaded_dll): Likewise. * event-loop.c (append_callback_event): Likewise. (create_file_handler): Likewise. (create_file_event): Likewise. * hostio.c (handle_open): Likewise. * inferiors.c (add_thread): Likewise. (add_process): Likewise. * linux-aarch64-low.c (aarch64_linux_new_process): Likewise. * linux-arm-low.c (arm_new_process): Likewise. (arm_new_thread): Likewise. * linux-low.c (add_to_pid_list): Likewise. (linux_add_process): Likewise. (handle_extended_wait): Likewise. (add_lwp): Likewise. (enqueue_one_deferred_signal): Likewise. (enqueue_pending_signal): Likewise. (linux_resume_one_lwp_throw): Likewise. (linux_resume_one_thread): Likewise. (linux_read_memory): Likewise. (linux_write_memory): Likewise. * linux-mips-low.c (mips_linux_new_process): Likewise. (mips_linux_new_thread): Likewise. (mips_add_watchpoint): Likewise. * linux-x86-low.c (initialize_low_arch): Likewise. * lynx-low.c (lynx_add_process): Likewise. * mem-break.c (set_raw_breakpoint_at): Likewise. (set_breakpoint): Likewise. (add_condition_to_breakpoint): Likewise. (add_commands_to_breakpoint): Likewise. (clone_agent_expr): Likewise. (clone_one_breakpoint): Likewise. * regcache.c (new_register_cache): Likewise. * remote-utils.c (look_up_one_symbol): Likewise. * server.c (queue_stop_reply): Likewise. (start_inferior): Likewise. (queue_stop_reply_callback): Likewise. (handle_target_event): Likewise. * spu-low.c (fetch_ppc_memory): Likewise. (store_ppc_memory): Likewise. * target.c (set_target_ops): Likewise. * thread-db.c (thread_db_load_search): Likewise. (try_thread_db_load_1): Likewise. * tracepoint.c (add_tracepoint): Likewise. (add_tracepoint_action): Likewise. (create_trace_state_variable): Likewise. (cmd_qtdpsrc): Likewise. (cmd_qtro): Likewise. (add_while_stepping_state): Likewise. * win32-low.c (child_add_thread): Likewise. (get_image_name): Likewise.
686 lines
17 KiB
C
686 lines
17 KiB
C
/* Low level interface to SPUs, for the remote server for GDB.
|
|
Copyright (C) 2006-2015 Free Software Foundation, Inc.
|
|
|
|
Contributed by Ulrich Weigand <uweigand@de.ibm.com>.
|
|
|
|
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 "server.h"
|
|
|
|
#include "gdb_wait.h"
|
|
#include <sys/ptrace.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <sys/syscall.h>
|
|
#include "filestuff.h"
|
|
#include "hostio.h"
|
|
|
|
/* Some older glibc versions do not define this. */
|
|
#ifndef __WNOTHREAD
|
|
#define __WNOTHREAD 0x20000000 /* Don't wait on children of other
|
|
threads in this group */
|
|
#endif
|
|
|
|
#define PTRACE_TYPE_RET long
|
|
#define PTRACE_TYPE_ARG3 long
|
|
|
|
/* Number of registers. */
|
|
#define SPU_NUM_REGS 130
|
|
#define SPU_NUM_CORE_REGS 128
|
|
|
|
/* Special registers. */
|
|
#define SPU_ID_REGNUM 128
|
|
#define SPU_PC_REGNUM 129
|
|
|
|
/* PPU side system calls. */
|
|
#define INSTR_SC 0x44000002
|
|
#define NR_spu_run 0x0116
|
|
|
|
/* These are used in remote-utils.c. */
|
|
int using_threads = 0;
|
|
|
|
/* Defined in auto-generated file reg-spu.c. */
|
|
void init_registers_spu (void);
|
|
extern const struct target_desc *tdesc_spu;
|
|
|
|
/* Fetch PPU register REGNO. */
|
|
static CORE_ADDR
|
|
fetch_ppc_register (int regno)
|
|
{
|
|
PTRACE_TYPE_RET res;
|
|
|
|
int tid = ptid_get_lwp (current_ptid);
|
|
|
|
#ifndef __powerpc64__
|
|
/* If running as a 32-bit process on a 64-bit system, we attempt
|
|
to get the full 64-bit register content of the target process.
|
|
If the PPC special ptrace call fails, we're on a 32-bit system;
|
|
just fall through to the regular ptrace call in that case. */
|
|
{
|
|
char buf[8];
|
|
|
|
errno = 0;
|
|
ptrace (PPC_PTRACE_PEEKUSR_3264, tid,
|
|
(PTRACE_TYPE_ARG3) (regno * 8), buf);
|
|
if (errno == 0)
|
|
ptrace (PPC_PTRACE_PEEKUSR_3264, tid,
|
|
(PTRACE_TYPE_ARG3) (regno * 8 + 4), buf + 4);
|
|
if (errno == 0)
|
|
return (CORE_ADDR) *(unsigned long long *)buf;
|
|
}
|
|
#endif
|
|
|
|
errno = 0;
|
|
res = ptrace (PT_READ_U, tid,
|
|
(PTRACE_TYPE_ARG3) (regno * sizeof (PTRACE_TYPE_RET)), 0);
|
|
if (errno != 0)
|
|
{
|
|
char mess[128];
|
|
sprintf (mess, "reading PPC register #%d", regno);
|
|
perror_with_name (mess);
|
|
}
|
|
|
|
return (CORE_ADDR) (unsigned long) res;
|
|
}
|
|
|
|
/* Fetch WORD from PPU memory at (aligned) MEMADDR in thread TID. */
|
|
static int
|
|
fetch_ppc_memory_1 (int tid, CORE_ADDR memaddr, PTRACE_TYPE_RET *word)
|
|
{
|
|
errno = 0;
|
|
|
|
#ifndef __powerpc64__
|
|
if (memaddr >> 32)
|
|
{
|
|
unsigned long long addr_8 = (unsigned long long) memaddr;
|
|
ptrace (PPC_PTRACE_PEEKTEXT_3264, tid, (PTRACE_TYPE_ARG3) &addr_8, word);
|
|
}
|
|
else
|
|
#endif
|
|
*word = ptrace (PT_READ_I, tid, (PTRACE_TYPE_ARG3) (size_t) memaddr, 0);
|
|
|
|
return errno;
|
|
}
|
|
|
|
/* Store WORD into PPU memory at (aligned) MEMADDR in thread TID. */
|
|
static int
|
|
store_ppc_memory_1 (int tid, CORE_ADDR memaddr, PTRACE_TYPE_RET word)
|
|
{
|
|
errno = 0;
|
|
|
|
#ifndef __powerpc64__
|
|
if (memaddr >> 32)
|
|
{
|
|
unsigned long long addr_8 = (unsigned long long) memaddr;
|
|
ptrace (PPC_PTRACE_POKEDATA_3264, tid, (PTRACE_TYPE_ARG3) &addr_8, word);
|
|
}
|
|
else
|
|
#endif
|
|
ptrace (PT_WRITE_D, tid, (PTRACE_TYPE_ARG3) (size_t) memaddr, word);
|
|
|
|
return errno;
|
|
}
|
|
|
|
/* Fetch LEN bytes of PPU memory at MEMADDR to MYADDR. */
|
|
static int
|
|
fetch_ppc_memory (CORE_ADDR memaddr, char *myaddr, int len)
|
|
{
|
|
int i, ret;
|
|
|
|
CORE_ADDR addr = memaddr & -(CORE_ADDR) sizeof (PTRACE_TYPE_RET);
|
|
int count = ((((memaddr + len) - addr) + sizeof (PTRACE_TYPE_RET) - 1)
|
|
/ sizeof (PTRACE_TYPE_RET));
|
|
PTRACE_TYPE_RET *buffer;
|
|
|
|
int tid = ptid_get_lwp (current_ptid);
|
|
|
|
buffer = XALLOCAVEC (PTRACE_TYPE_RET, count);
|
|
for (i = 0; i < count; i++, addr += sizeof (PTRACE_TYPE_RET))
|
|
if ((ret = fetch_ppc_memory_1 (tid, addr, &buffer[i])) != 0)
|
|
return ret;
|
|
|
|
memcpy (myaddr,
|
|
(char *) buffer + (memaddr & (sizeof (PTRACE_TYPE_RET) - 1)),
|
|
len);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* Store LEN bytes from MYADDR to PPU memory at MEMADDR. */
|
|
static int
|
|
store_ppc_memory (CORE_ADDR memaddr, char *myaddr, int len)
|
|
{
|
|
int i, ret;
|
|
|
|
CORE_ADDR addr = memaddr & -(CORE_ADDR) sizeof (PTRACE_TYPE_RET);
|
|
int count = ((((memaddr + len) - addr) + sizeof (PTRACE_TYPE_RET) - 1)
|
|
/ sizeof (PTRACE_TYPE_RET));
|
|
PTRACE_TYPE_RET *buffer;
|
|
|
|
int tid = ptid_get_lwp (current_ptid);
|
|
|
|
buffer = XALLOCAVEC (PTRACE_TYPE_RET, count);
|
|
|
|
if (addr != memaddr || len < (int) sizeof (PTRACE_TYPE_RET))
|
|
if ((ret = fetch_ppc_memory_1 (tid, addr, &buffer[0])) != 0)
|
|
return ret;
|
|
|
|
if (count > 1)
|
|
if ((ret = fetch_ppc_memory_1 (tid, addr + (count - 1)
|
|
* sizeof (PTRACE_TYPE_RET),
|
|
&buffer[count - 1])) != 0)
|
|
return ret;
|
|
|
|
memcpy ((char *) buffer + (memaddr & (sizeof (PTRACE_TYPE_RET) - 1)),
|
|
myaddr, len);
|
|
|
|
for (i = 0; i < count; i++, addr += sizeof (PTRACE_TYPE_RET))
|
|
if ((ret = store_ppc_memory_1 (tid, addr, buffer[i])) != 0)
|
|
return ret;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* If the PPU thread is currently stopped on a spu_run system call,
|
|
return to FD and ADDR the file handle and NPC parameter address
|
|
used with the system call. Return non-zero if successful. */
|
|
static int
|
|
parse_spufs_run (int *fd, CORE_ADDR *addr)
|
|
{
|
|
unsigned int insn;
|
|
CORE_ADDR pc = fetch_ppc_register (32); /* nip */
|
|
|
|
/* Fetch instruction preceding current NIP. */
|
|
if (fetch_ppc_memory (pc-4, (char *) &insn, 4) != 0)
|
|
return 0;
|
|
/* It should be a "sc" instruction. */
|
|
if (insn != INSTR_SC)
|
|
return 0;
|
|
/* System call number should be NR_spu_run. */
|
|
if (fetch_ppc_register (0) != NR_spu_run)
|
|
return 0;
|
|
|
|
/* Register 3 contains fd, register 4 the NPC param pointer. */
|
|
*fd = fetch_ppc_register (34); /* orig_gpr3 */
|
|
*addr = fetch_ppc_register (4);
|
|
return 1;
|
|
}
|
|
|
|
|
|
/* Copy LEN bytes at OFFSET in spufs file ANNEX into/from READBUF or WRITEBUF,
|
|
using the /proc file system. */
|
|
static int
|
|
spu_proc_xfer_spu (const char *annex, unsigned char *readbuf,
|
|
const unsigned char *writebuf,
|
|
CORE_ADDR offset, int len)
|
|
{
|
|
char buf[128];
|
|
int fd = 0;
|
|
int ret = -1;
|
|
|
|
if (!annex)
|
|
return 0;
|
|
|
|
sprintf (buf, "/proc/%ld/fd/%s", ptid_get_lwp (current_ptid), annex);
|
|
fd = open (buf, writebuf? O_WRONLY : O_RDONLY);
|
|
if (fd <= 0)
|
|
return -1;
|
|
|
|
if (offset != 0
|
|
&& lseek (fd, (off_t) offset, SEEK_SET) != (off_t) offset)
|
|
{
|
|
close (fd);
|
|
return 0;
|
|
}
|
|
|
|
if (writebuf)
|
|
ret = write (fd, writebuf, (size_t) len);
|
|
else if (readbuf)
|
|
ret = read (fd, readbuf, (size_t) len);
|
|
|
|
close (fd);
|
|
return ret;
|
|
}
|
|
|
|
|
|
/* Start an inferior process and returns its pid.
|
|
ALLARGS is a vector of program-name and args. */
|
|
static int
|
|
spu_create_inferior (char *program, char **allargs)
|
|
{
|
|
int pid;
|
|
ptid_t ptid;
|
|
struct process_info *proc;
|
|
|
|
pid = fork ();
|
|
if (pid < 0)
|
|
perror_with_name ("fork");
|
|
|
|
if (pid == 0)
|
|
{
|
|
close_most_fds ();
|
|
ptrace (PTRACE_TRACEME, 0, 0, 0);
|
|
|
|
setpgid (0, 0);
|
|
|
|
execv (program, allargs);
|
|
if (errno == ENOENT)
|
|
execvp (program, allargs);
|
|
|
|
fprintf (stderr, "Cannot exec %s: %s.\n", program,
|
|
strerror (errno));
|
|
fflush (stderr);
|
|
_exit (0177);
|
|
}
|
|
|
|
proc = add_process (pid, 0);
|
|
proc->tdesc = tdesc_spu;
|
|
|
|
ptid = ptid_build (pid, pid, 0);
|
|
add_thread (ptid, NULL);
|
|
return pid;
|
|
}
|
|
|
|
/* Attach to an inferior process. */
|
|
int
|
|
spu_attach (unsigned long pid)
|
|
{
|
|
ptid_t ptid;
|
|
struct process_info *proc;
|
|
|
|
if (ptrace (PTRACE_ATTACH, pid, 0, 0) != 0)
|
|
{
|
|
fprintf (stderr, "Cannot attach to process %ld: %s (%d)\n", pid,
|
|
strerror (errno), errno);
|
|
fflush (stderr);
|
|
_exit (0177);
|
|
}
|
|
|
|
proc = add_process (pid, 1);
|
|
proc->tdesc = tdesc_spu;
|
|
ptid = ptid_build (pid, pid, 0);
|
|
add_thread (ptid, NULL);
|
|
return 0;
|
|
}
|
|
|
|
/* Kill the inferior process. */
|
|
static int
|
|
spu_kill (int pid)
|
|
{
|
|
int status, ret;
|
|
struct process_info *process = find_process_pid (pid);
|
|
if (process == NULL)
|
|
return -1;
|
|
|
|
ptrace (PTRACE_KILL, pid, 0, 0);
|
|
|
|
do {
|
|
ret = waitpid (pid, &status, 0);
|
|
if (WIFEXITED (status) || WIFSIGNALED (status))
|
|
break;
|
|
} while (ret != -1 || errno != ECHILD);
|
|
|
|
clear_inferiors ();
|
|
remove_process (process);
|
|
return 0;
|
|
}
|
|
|
|
/* Detach from inferior process. */
|
|
static int
|
|
spu_detach (int pid)
|
|
{
|
|
struct process_info *process = find_process_pid (pid);
|
|
if (process == NULL)
|
|
return -1;
|
|
|
|
ptrace (PTRACE_DETACH, pid, 0, 0);
|
|
|
|
clear_inferiors ();
|
|
remove_process (process);
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
spu_mourn (struct process_info *process)
|
|
{
|
|
remove_process (process);
|
|
}
|
|
|
|
static void
|
|
spu_join (int pid)
|
|
{
|
|
int status, ret;
|
|
|
|
do {
|
|
ret = waitpid (pid, &status, 0);
|
|
if (WIFEXITED (status) || WIFSIGNALED (status))
|
|
break;
|
|
} while (ret != -1 || errno != ECHILD);
|
|
}
|
|
|
|
/* Return nonzero if the given thread is still alive. */
|
|
static int
|
|
spu_thread_alive (ptid_t ptid)
|
|
{
|
|
return ptid_equal (ptid, current_ptid);
|
|
}
|
|
|
|
/* Resume process. */
|
|
static void
|
|
spu_resume (struct thread_resume *resume_info, size_t n)
|
|
{
|
|
struct thread_info *thr = get_first_thread ();
|
|
size_t i;
|
|
|
|
for (i = 0; i < n; i++)
|
|
if (ptid_equal (resume_info[i].thread, minus_one_ptid)
|
|
|| ptid_equal (resume_info[i].thread, ptid_of (thr)))
|
|
break;
|
|
|
|
if (i == n)
|
|
return;
|
|
|
|
/* We don't support hardware single-stepping right now, assume
|
|
GDB knows to use software single-stepping. */
|
|
if (resume_info[i].kind == resume_step)
|
|
fprintf (stderr, "Hardware single-step not supported.\n");
|
|
|
|
regcache_invalidate ();
|
|
|
|
errno = 0;
|
|
ptrace (PTRACE_CONT, ptid_get_lwp (ptid_of (thr)), 0, resume_info[i].sig);
|
|
if (errno)
|
|
perror_with_name ("ptrace");
|
|
}
|
|
|
|
/* Wait for process, returns status. */
|
|
static ptid_t
|
|
spu_wait (ptid_t ptid, struct target_waitstatus *ourstatus, int options)
|
|
{
|
|
int pid = ptid_get_pid (ptid);
|
|
int w;
|
|
int ret;
|
|
|
|
while (1)
|
|
{
|
|
ret = waitpid (pid, &w, WNOHANG | __WALL | __WNOTHREAD);
|
|
|
|
if (ret == -1)
|
|
{
|
|
if (errno != ECHILD)
|
|
perror_with_name ("waitpid");
|
|
}
|
|
else if (ret > 0)
|
|
break;
|
|
|
|
usleep (1000);
|
|
}
|
|
|
|
/* On the first wait, continue running the inferior until we are
|
|
blocked inside an spu_run system call. */
|
|
if (!server_waiting)
|
|
{
|
|
int fd;
|
|
CORE_ADDR addr;
|
|
|
|
while (!parse_spufs_run (&fd, &addr))
|
|
{
|
|
ptrace (PT_SYSCALL, pid, (PTRACE_TYPE_ARG3) 0, 0);
|
|
waitpid (pid, NULL, __WALL | __WNOTHREAD);
|
|
}
|
|
}
|
|
|
|
if (WIFEXITED (w))
|
|
{
|
|
fprintf (stderr, "\nChild exited with retcode = %x \n", WEXITSTATUS (w));
|
|
ourstatus->kind = TARGET_WAITKIND_EXITED;
|
|
ourstatus->value.integer = WEXITSTATUS (w);
|
|
clear_inferiors ();
|
|
return pid_to_ptid (ret);
|
|
}
|
|
else if (!WIFSTOPPED (w))
|
|
{
|
|
fprintf (stderr, "\nChild terminated with signal = %x \n", WTERMSIG (w));
|
|
ourstatus->kind = TARGET_WAITKIND_SIGNALLED;
|
|
ourstatus->value.sig = gdb_signal_from_host (WTERMSIG (w));
|
|
clear_inferiors ();
|
|
return pid_to_ptid (ret);
|
|
}
|
|
|
|
/* After attach, we may have received a SIGSTOP. Do not return this
|
|
as signal to GDB, or else it will try to continue with SIGSTOP ... */
|
|
if (!server_waiting)
|
|
{
|
|
ourstatus->kind = TARGET_WAITKIND_STOPPED;
|
|
ourstatus->value.sig = GDB_SIGNAL_0;
|
|
return ptid_build (ret, ret, 0);
|
|
}
|
|
|
|
ourstatus->kind = TARGET_WAITKIND_STOPPED;
|
|
ourstatus->value.sig = gdb_signal_from_host (WSTOPSIG (w));
|
|
return ptid_build (ret, ret, 0);
|
|
}
|
|
|
|
/* Fetch inferior registers. */
|
|
static void
|
|
spu_fetch_registers (struct regcache *regcache, int regno)
|
|
{
|
|
int fd;
|
|
CORE_ADDR addr;
|
|
|
|
/* We must be stopped on a spu_run system call. */
|
|
if (!parse_spufs_run (&fd, &addr))
|
|
return;
|
|
|
|
/* The ID register holds the spufs file handle. */
|
|
if (regno == -1 || regno == SPU_ID_REGNUM)
|
|
supply_register (regcache, SPU_ID_REGNUM, (char *)&fd);
|
|
|
|
/* The NPC register is found at ADDR. */
|
|
if (regno == -1 || regno == SPU_PC_REGNUM)
|
|
{
|
|
char buf[4];
|
|
if (fetch_ppc_memory (addr, buf, 4) == 0)
|
|
supply_register (regcache, SPU_PC_REGNUM, buf);
|
|
}
|
|
|
|
/* The GPRs are found in the "regs" spufs file. */
|
|
if (regno == -1 || (regno >= 0 && regno < SPU_NUM_CORE_REGS))
|
|
{
|
|
unsigned char buf[16*SPU_NUM_CORE_REGS];
|
|
char annex[32];
|
|
int i;
|
|
|
|
sprintf (annex, "%d/regs", fd);
|
|
if (spu_proc_xfer_spu (annex, buf, NULL, 0, sizeof buf) == sizeof buf)
|
|
for (i = 0; i < SPU_NUM_CORE_REGS; i++)
|
|
supply_register (regcache, i, buf + i*16);
|
|
}
|
|
}
|
|
|
|
/* Store inferior registers. */
|
|
static void
|
|
spu_store_registers (struct regcache *regcache, int regno)
|
|
{
|
|
int fd;
|
|
CORE_ADDR addr;
|
|
|
|
/* ??? Some callers use 0 to mean all registers. */
|
|
if (regno == 0)
|
|
regno = -1;
|
|
|
|
/* We must be stopped on a spu_run system call. */
|
|
if (!parse_spufs_run (&fd, &addr))
|
|
return;
|
|
|
|
/* The NPC register is found at ADDR. */
|
|
if (regno == -1 || regno == SPU_PC_REGNUM)
|
|
{
|
|
char buf[4];
|
|
collect_register (regcache, SPU_PC_REGNUM, buf);
|
|
store_ppc_memory (addr, buf, 4);
|
|
}
|
|
|
|
/* The GPRs are found in the "regs" spufs file. */
|
|
if (regno == -1 || (regno >= 0 && regno < SPU_NUM_CORE_REGS))
|
|
{
|
|
unsigned char buf[16*SPU_NUM_CORE_REGS];
|
|
char annex[32];
|
|
int i;
|
|
|
|
for (i = 0; i < SPU_NUM_CORE_REGS; i++)
|
|
collect_register (regcache, i, buf + i*16);
|
|
|
|
sprintf (annex, "%d/regs", fd);
|
|
spu_proc_xfer_spu (annex, NULL, buf, 0, sizeof buf);
|
|
}
|
|
}
|
|
|
|
/* Copy LEN bytes from inferior's memory starting at MEMADDR
|
|
to debugger memory starting at MYADDR. */
|
|
static int
|
|
spu_read_memory (CORE_ADDR memaddr, unsigned char *myaddr, int len)
|
|
{
|
|
int fd, ret;
|
|
CORE_ADDR addr;
|
|
char annex[32], lslr_annex[32], buf[32];
|
|
CORE_ADDR lslr;
|
|
|
|
/* We must be stopped on a spu_run system call. */
|
|
if (!parse_spufs_run (&fd, &addr))
|
|
return 0;
|
|
|
|
/* Use the "mem" spufs file to access SPU local store. */
|
|
sprintf (annex, "%d/mem", fd);
|
|
ret = spu_proc_xfer_spu (annex, myaddr, NULL, memaddr, len);
|
|
if (ret > 0)
|
|
return ret == len ? 0 : EIO;
|
|
|
|
/* SPU local store access wraps the address around at the
|
|
local store limit. We emulate this here. To avoid needing
|
|
an extra access to retrieve the LSLR, we only do that after
|
|
trying the original address first, and getting end-of-file. */
|
|
sprintf (lslr_annex, "%d/lslr", fd);
|
|
memset (buf, 0, sizeof buf);
|
|
if (spu_proc_xfer_spu (lslr_annex, (unsigned char *)buf, NULL,
|
|
0, sizeof buf) <= 0)
|
|
return ret;
|
|
|
|
lslr = strtoul (buf, NULL, 16);
|
|
ret = spu_proc_xfer_spu (annex, myaddr, NULL, memaddr & lslr, len);
|
|
|
|
return ret == len ? 0 : EIO;
|
|
}
|
|
|
|
/* Copy LEN bytes of data from debugger memory at MYADDR
|
|
to inferior's memory at MEMADDR.
|
|
On failure (cannot write the inferior)
|
|
returns the value of errno. */
|
|
static int
|
|
spu_write_memory (CORE_ADDR memaddr, const unsigned char *myaddr, int len)
|
|
{
|
|
int fd, ret;
|
|
CORE_ADDR addr;
|
|
char annex[32], lslr_annex[32], buf[32];
|
|
CORE_ADDR lslr;
|
|
|
|
/* We must be stopped on a spu_run system call. */
|
|
if (!parse_spufs_run (&fd, &addr))
|
|
return 0;
|
|
|
|
/* Use the "mem" spufs file to access SPU local store. */
|
|
sprintf (annex, "%d/mem", fd);
|
|
ret = spu_proc_xfer_spu (annex, NULL, myaddr, memaddr, len);
|
|
if (ret > 0)
|
|
return ret == len ? 0 : EIO;
|
|
|
|
/* SPU local store access wraps the address around at the
|
|
local store limit. We emulate this here. To avoid needing
|
|
an extra access to retrieve the LSLR, we only do that after
|
|
trying the original address first, and getting end-of-file. */
|
|
sprintf (lslr_annex, "%d/lslr", fd);
|
|
memset (buf, 0, sizeof buf);
|
|
if (spu_proc_xfer_spu (lslr_annex, (unsigned char *)buf, NULL,
|
|
0, sizeof buf) <= 0)
|
|
return ret;
|
|
|
|
lslr = strtoul (buf, NULL, 16);
|
|
ret = spu_proc_xfer_spu (annex, NULL, myaddr, memaddr & lslr, len);
|
|
|
|
return ret == len ? 0 : EIO;
|
|
}
|
|
|
|
/* Look up special symbols -- unneded here. */
|
|
static void
|
|
spu_look_up_symbols (void)
|
|
{
|
|
}
|
|
|
|
/* Send signal to inferior. */
|
|
static void
|
|
spu_request_interrupt (void)
|
|
{
|
|
struct thread_info *thr = get_first_thread ();
|
|
|
|
syscall (SYS_tkill, lwpid_of (thr), SIGINT);
|
|
}
|
|
|
|
static struct target_ops spu_target_ops = {
|
|
spu_create_inferior,
|
|
NULL, /* arch_setup */
|
|
spu_attach,
|
|
spu_kill,
|
|
spu_detach,
|
|
spu_mourn,
|
|
spu_join,
|
|
spu_thread_alive,
|
|
spu_resume,
|
|
spu_wait,
|
|
spu_fetch_registers,
|
|
spu_store_registers,
|
|
NULL, /* prepare_to_access_memory */
|
|
NULL, /* done_accessing_memory */
|
|
spu_read_memory,
|
|
spu_write_memory,
|
|
spu_look_up_symbols,
|
|
spu_request_interrupt,
|
|
NULL,
|
|
NULL, /* supports_z_point_type */
|
|
NULL,
|
|
NULL,
|
|
NULL, /* stopped_by_sw_breakpoint */
|
|
NULL, /* supports_stopped_by_sw_breakpoint */
|
|
NULL, /* stopped_by_hw_breakpoint */
|
|
NULL, /* supports_stopped_by_hw_breakpoint */
|
|
NULL, /* supports_conditional_breakpoints */
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
spu_proc_xfer_spu,
|
|
hostio_last_error_from_errno,
|
|
};
|
|
|
|
void
|
|
initialize_low (void)
|
|
{
|
|
static const unsigned char breakpoint[] = { 0x00, 0x00, 0x3f, 0xff };
|
|
|
|
set_target_ops (&spu_target_ops);
|
|
set_breakpoint_data (breakpoint, sizeof breakpoint);
|
|
init_registers_spu ();
|
|
}
|