2003-10-10 15:13:11 +08:00
|
|
|
/* Target-dependent code for Renesas D10V, for GDB.
|
2002-03-19 10:51:09 +08:00
|
|
|
|
2003-01-05 09:39:56 +08:00
|
|
|
Copyright 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003 Free Software
|
2002-03-19 10:51:09 +08:00
|
|
|
Foundation, Inc.
|
1999-04-16 09:35:26 +08:00
|
|
|
|
1999-07-08 04:19:36 +08:00
|
|
|
This file is part of GDB.
|
1999-04-16 09:35:26 +08:00
|
|
|
|
1999-07-08 04:19:36 +08:00
|
|
|
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 2 of the License, or
|
|
|
|
(at your option) any later version.
|
1999-04-16 09:35:26 +08:00
|
|
|
|
1999-07-08 04:19:36 +08:00
|
|
|
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.
|
1999-04-16 09:35:26 +08:00
|
|
|
|
1999-07-08 04:19:36 +08:00
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 59 Temple Place - Suite 330,
|
|
|
|
Boston, MA 02111-1307, USA. */
|
1999-04-16 09:35:26 +08:00
|
|
|
|
|
|
|
/* Contributed by Martin Hunt, hunt@cygnus.com */
|
|
|
|
|
|
|
|
#include "defs.h"
|
|
|
|
#include "frame.h"
|
2003-01-30 23:11:20 +08:00
|
|
|
#include "frame-unwind.h"
|
2003-04-02 03:55:03 +08:00
|
|
|
#include "frame-base.h"
|
1999-04-16 09:35:26 +08:00
|
|
|
#include "symtab.h"
|
|
|
|
#include "gdbtypes.h"
|
|
|
|
#include "gdbcmd.h"
|
|
|
|
#include "gdbcore.h"
|
|
|
|
#include "gdb_string.h"
|
|
|
|
#include "value.h"
|
|
|
|
#include "inferior.h"
|
1999-07-08 04:19:36 +08:00
|
|
|
#include "dis-asm.h"
|
1999-04-16 09:35:26 +08:00
|
|
|
#include "symfile.h"
|
|
|
|
#include "objfiles.h"
|
1999-08-31 09:14:27 +08:00
|
|
|
#include "language.h"
|
2000-04-27 12:25:45 +08:00
|
|
|
#include "arch-utils.h"
|
2001-03-01 09:39:22 +08:00
|
|
|
#include "regcache.h"
|
2003-03-29 09:40:01 +08:00
|
|
|
#include "remote.h"
|
2000-06-02 09:59:13 +08:00
|
|
|
#include "floatformat.h"
|
2002-05-24 08:12:18 +08:00
|
|
|
#include "gdb/sim-d10v.h"
|
2002-05-29 09:51:17 +08:00
|
|
|
#include "sim-regno.h"
|
2003-05-04 03:13:04 +08:00
|
|
|
#include "disasm.h"
|
2003-06-09 00:09:46 +08:00
|
|
|
#include "trad-frame.h"
|
1999-11-17 10:31:06 +08:00
|
|
|
|
2003-01-13 02:59:53 +08:00
|
|
|
#include "gdb_assert.h"
|
|
|
|
|
1999-11-17 10:31:06 +08:00
|
|
|
struct gdbarch_tdep
|
|
|
|
{
|
|
|
|
int a0_regnum;
|
|
|
|
int nr_dmap_regs;
|
2003-05-08 03:21:13 +08:00
|
|
|
unsigned long (*dmap_register) (void *regcache, int nr);
|
|
|
|
unsigned long (*imap_register) (void *regcache, int nr);
|
1999-11-17 10:31:06 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/* These are the addresses the D10V-EVA board maps data and
|
2003-10-10 08:25:43 +08:00
|
|
|
instruction memory to. */
|
1999-06-08 03:19:32 +08:00
|
|
|
|
2002-05-23 23:52:01 +08:00
|
|
|
enum memspace {
|
|
|
|
DMEM_START = 0x2000000,
|
|
|
|
IMEM_START = 0x1000000,
|
|
|
|
STACK_START = 0x200bffe
|
|
|
|
};
|
1999-06-08 03:19:32 +08:00
|
|
|
|
2003-10-10 08:25:43 +08:00
|
|
|
/* d10v register names. */
|
1999-11-17 10:31:06 +08:00
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
R0_REGNUM = 0,
|
2002-05-23 23:52:01 +08:00
|
|
|
R3_REGNUM = 3,
|
2003-03-31 10:36:59 +08:00
|
|
|
D10V_FP_REGNUM = 11,
|
1999-11-17 10:31:06 +08:00
|
|
|
LR_REGNUM = 13,
|
2003-04-26 05:20:58 +08:00
|
|
|
D10V_SP_REGNUM = 15,
|
1999-11-17 10:31:06 +08:00
|
|
|
PSW_REGNUM = 16,
|
2003-05-03 07:56:12 +08:00
|
|
|
D10V_PC_REGNUM = 18,
|
1999-11-17 10:31:06 +08:00
|
|
|
NR_IMAP_REGS = 2,
|
2002-05-23 23:52:01 +08:00
|
|
|
NR_A_REGS = 2,
|
|
|
|
TS2_NUM_REGS = 37,
|
|
|
|
TS3_NUM_REGS = 42,
|
2003-10-10 08:25:43 +08:00
|
|
|
/* d10v calling convention. */
|
2002-05-23 23:52:01 +08:00
|
|
|
ARG1_REGNUM = R0_REGNUM,
|
2003-11-09 22:20:55 +08:00
|
|
|
ARGN_REGNUM = R3_REGNUM
|
1999-11-17 10:31:06 +08:00
|
|
|
};
|
2002-05-23 23:52:01 +08:00
|
|
|
|
2003-06-08 Andrew Cagney <cagney@redhat.com>
* acinclude.m4 (gcc_AC_CHECK_DECL, (gcc_AC_CHECK_DECL): Stolen
from GCC's acinclude.m4.
* configure.in: Check for getopt's delcaration.
* aclocal.m4, config.in, configure: Re-generate.
* main.c (error_init): Delete declaration.
* defs.h (error_init): Declare.
* rs6000-tdep.c (rs6000_fetch_pointer_argument): Make static.
(rs6000_convert_from_func_ptr_addr): Make static.
(_initialize_rs6000_tdep): Add declaration.
* cli/cli-cmds.c (dont_repeat): Delete declaration.
(show_commands, set_verbose, show_history): Delete declaration.
* top.h (set_verbose): Add declaration.
(show_history, set_history, show_commands): Add declaration.
(do_restore_instream_cleanup): Add declaration.
* objc-lang.c (specialcmp): Make static.
(print_object_command): Make static.
(find_objc_msgsend): Make static.
(find_objc_msgcall_submethod_helper): Make static.
(find_objc_msgcall_submethod): Make static.
(_initialize_objc_language): Add declaration.
(find_implementation_from_class): Make static.
(find_implementation): Make static.
* objc-exp.y (yylex): Delete lookup_struct_typedef declaration.
* objc-lang.h (lookup_struct_typedef): Add declaration.
* cli/cli-interp.c (_initialize_cli_interp): Add declaration.
* cli/cli-script.c (clear_hook_in_cleanup): Make static.
(do_restore_user_call_depth): Make static.
(do_restore_instream_cleanup): Delete declaration.
(dont_repeat): Delete declaration.
* cli/cli-decode.c (add_abbrev_cmd): Delete function.
* cli/cli-dump.c (_initialize_cli_dump): Add declaration.
* reggroups.c (_initialize_reggroup): Add declaration.
* cp-support.c (_initialize_cp_support): Add declaration.
* cp-abi.c (_initialize_cp_abi): Add declaration.
* hpacc-abi.c (_initialize_hpacc_abi): Add declaration.
* gnu-v3-abi.c (gnuv3_baseclass_offset): Make static.
(_initialize_gnu_v3_abi): Add declaration.
* gnu-v2-abi.c (gnuv2_value_rtti_type): Make static.
(_initialize_gnu_v2_abi): Add declaration.
* frame-base.c (_initialize_frame_base): Add declaration.
* doublest.c (floatformat_from_length): Make static.
* frame-unwind.c (_initialize_frame_unwind): Add declaration.
* frame.c (create_sentinel_frame): Make static.
(_initialize_frame): Add declaration.
* top.c (do_catch_errors): Make static.
(gdb_rl_operate_and_get_next_completion): Make static.
* typeprint.c: Include "typeprint.h".
* sentinel-frame.c (sentinel_frame_prev_register): Make static.
(sentinel_frame_this_id): Make static.
* p-valprint.c (_initialize_pascal_valprint): Add declaration.
* ui-out.c (make_cleanup_ui_out_begin_end): Delete function.
* dwarf2-frame.c (dwarf2_frame_cache): Make static.
* p-exp.y (push_current_type, pop_current_type): ISO C declaration.
* dwarf2expr.h (dwarf_expr_context): ISO C declaration.
* maint.c (maintenance_print_architecture): Make static.
* signals/signals.c (_initialize_signals): Add declaration.
* std-regs.c (_initialize_frame_reg): Add declaration.
* jv-exp.y (push_variable): ISO C definition.
(push_qualified_expression_name): Ditto.
* memattr.c (_initialize_mem): Add declaration.
* remote.c (remote_check_watch_resources): Make static.
(remote_stopped_by_watchpoint): Make static.
(remote_stopped_data_address): Make static.
* d10v-tdep.c (nr_dmap_regs): Make static.
(a0_regnum): Make static.
(d10v_frame_unwind_cache): Make static.
(d10v_frame_p): Make static.
* osabi.c (show_osabi): Make static.
(_initialize_gdb_osabi): Add extern declaration.
* gdbtypes.c (make_qualified_type): Make static.
(safe_parse_type): Make static.
* macrocmd.c (_initialize_macrocmd): Add extern declaration.
* macrotab.c (macro_bcache_free): Make static.
* interps.c (interp_set_quiet): Make static.
(interpreter_exec_cmd): Make static.
* stack.h (select_frame_command): New file.
* stack.c: Include "stack.h".
(select_frame_command_wrapper): Delete function.
(select_frame_command): Make global.
* infcall.c: Include "infcall.h".
* linespec.c: Include "linespec.h".
* symfile.c (sections_overlap): Make static.
* cp-support.h (cp_initialize_namespace): ISO C declaration.
* charset.c (_initialize_charset): Add missing prototype.
* regcache.c (init_legacy_regcache_descr): Make static.
(do_regcache_xfree): Make static.
(regcache_xfer_part): Make static.
(_initialize_regcache): Add missing prototype.
* breakpoint.c (parse_breakpoint_sals): Make static.
(breakpoint_sals_to_pc): Make static.
* interps.h (clear_interpreter_hooks): ISO C declaration.
* Makefile.in (stack_h): Define.
(stack.o, typeprint.o, mi-main.o): Update dependencies.
(mi-cmd-stack.o, infcall.o, linespec.o): Update dependencies.
Index: mi/ChangeLog
2003-06-08 Andrew Cagney <cagney@redhat.com>
* mi-parse.c (_initialize_mi_parse): Delete function.
* mi-main.c: Include "mi-main.h".
* mi-interp.c (_initialize_mi_interp): Add declaration.
* mi-cmd-stack.c: Include "stack.h".
(select_frame_command_wrapper): Delete extern declaration.
(mi_cmd_stack_select_frame): Replace select_frame_command_wrapper
with select_frame_command.
2003-06-09 02:27:14 +08:00
|
|
|
static int
|
2003-04-26 05:20:58 +08:00
|
|
|
nr_dmap_regs (struct gdbarch *gdbarch)
|
|
|
|
{
|
|
|
|
return gdbarch_tdep (gdbarch)->nr_dmap_regs;
|
|
|
|
}
|
|
|
|
|
2003-06-08 Andrew Cagney <cagney@redhat.com>
* acinclude.m4 (gcc_AC_CHECK_DECL, (gcc_AC_CHECK_DECL): Stolen
from GCC's acinclude.m4.
* configure.in: Check for getopt's delcaration.
* aclocal.m4, config.in, configure: Re-generate.
* main.c (error_init): Delete declaration.
* defs.h (error_init): Declare.
* rs6000-tdep.c (rs6000_fetch_pointer_argument): Make static.
(rs6000_convert_from_func_ptr_addr): Make static.
(_initialize_rs6000_tdep): Add declaration.
* cli/cli-cmds.c (dont_repeat): Delete declaration.
(show_commands, set_verbose, show_history): Delete declaration.
* top.h (set_verbose): Add declaration.
(show_history, set_history, show_commands): Add declaration.
(do_restore_instream_cleanup): Add declaration.
* objc-lang.c (specialcmp): Make static.
(print_object_command): Make static.
(find_objc_msgsend): Make static.
(find_objc_msgcall_submethod_helper): Make static.
(find_objc_msgcall_submethod): Make static.
(_initialize_objc_language): Add declaration.
(find_implementation_from_class): Make static.
(find_implementation): Make static.
* objc-exp.y (yylex): Delete lookup_struct_typedef declaration.
* objc-lang.h (lookup_struct_typedef): Add declaration.
* cli/cli-interp.c (_initialize_cli_interp): Add declaration.
* cli/cli-script.c (clear_hook_in_cleanup): Make static.
(do_restore_user_call_depth): Make static.
(do_restore_instream_cleanup): Delete declaration.
(dont_repeat): Delete declaration.
* cli/cli-decode.c (add_abbrev_cmd): Delete function.
* cli/cli-dump.c (_initialize_cli_dump): Add declaration.
* reggroups.c (_initialize_reggroup): Add declaration.
* cp-support.c (_initialize_cp_support): Add declaration.
* cp-abi.c (_initialize_cp_abi): Add declaration.
* hpacc-abi.c (_initialize_hpacc_abi): Add declaration.
* gnu-v3-abi.c (gnuv3_baseclass_offset): Make static.
(_initialize_gnu_v3_abi): Add declaration.
* gnu-v2-abi.c (gnuv2_value_rtti_type): Make static.
(_initialize_gnu_v2_abi): Add declaration.
* frame-base.c (_initialize_frame_base): Add declaration.
* doublest.c (floatformat_from_length): Make static.
* frame-unwind.c (_initialize_frame_unwind): Add declaration.
* frame.c (create_sentinel_frame): Make static.
(_initialize_frame): Add declaration.
* top.c (do_catch_errors): Make static.
(gdb_rl_operate_and_get_next_completion): Make static.
* typeprint.c: Include "typeprint.h".
* sentinel-frame.c (sentinel_frame_prev_register): Make static.
(sentinel_frame_this_id): Make static.
* p-valprint.c (_initialize_pascal_valprint): Add declaration.
* ui-out.c (make_cleanup_ui_out_begin_end): Delete function.
* dwarf2-frame.c (dwarf2_frame_cache): Make static.
* p-exp.y (push_current_type, pop_current_type): ISO C declaration.
* dwarf2expr.h (dwarf_expr_context): ISO C declaration.
* maint.c (maintenance_print_architecture): Make static.
* signals/signals.c (_initialize_signals): Add declaration.
* std-regs.c (_initialize_frame_reg): Add declaration.
* jv-exp.y (push_variable): ISO C definition.
(push_qualified_expression_name): Ditto.
* memattr.c (_initialize_mem): Add declaration.
* remote.c (remote_check_watch_resources): Make static.
(remote_stopped_by_watchpoint): Make static.
(remote_stopped_data_address): Make static.
* d10v-tdep.c (nr_dmap_regs): Make static.
(a0_regnum): Make static.
(d10v_frame_unwind_cache): Make static.
(d10v_frame_p): Make static.
* osabi.c (show_osabi): Make static.
(_initialize_gdb_osabi): Add extern declaration.
* gdbtypes.c (make_qualified_type): Make static.
(safe_parse_type): Make static.
* macrocmd.c (_initialize_macrocmd): Add extern declaration.
* macrotab.c (macro_bcache_free): Make static.
* interps.c (interp_set_quiet): Make static.
(interpreter_exec_cmd): Make static.
* stack.h (select_frame_command): New file.
* stack.c: Include "stack.h".
(select_frame_command_wrapper): Delete function.
(select_frame_command): Make global.
* infcall.c: Include "infcall.h".
* linespec.c: Include "linespec.h".
* symfile.c (sections_overlap): Make static.
* cp-support.h (cp_initialize_namespace): ISO C declaration.
* charset.c (_initialize_charset): Add missing prototype.
* regcache.c (init_legacy_regcache_descr): Make static.
(do_regcache_xfree): Make static.
(regcache_xfer_part): Make static.
(_initialize_regcache): Add missing prototype.
* breakpoint.c (parse_breakpoint_sals): Make static.
(breakpoint_sals_to_pc): Make static.
* interps.h (clear_interpreter_hooks): ISO C declaration.
* Makefile.in (stack_h): Define.
(stack.o, typeprint.o, mi-main.o): Update dependencies.
(mi-cmd-stack.o, infcall.o, linespec.o): Update dependencies.
Index: mi/ChangeLog
2003-06-08 Andrew Cagney <cagney@redhat.com>
* mi-parse.c (_initialize_mi_parse): Delete function.
* mi-main.c: Include "mi-main.h".
* mi-interp.c (_initialize_mi_interp): Add declaration.
* mi-cmd-stack.c: Include "stack.h".
(select_frame_command_wrapper): Delete extern declaration.
(mi_cmd_stack_select_frame): Replace select_frame_command_wrapper
with select_frame_command.
2003-06-09 02:27:14 +08:00
|
|
|
static int
|
2003-04-26 05:20:58 +08:00
|
|
|
a0_regnum (struct gdbarch *gdbarch)
|
|
|
|
{
|
|
|
|
return gdbarch_tdep (gdbarch)->a0_regnum;
|
|
|
|
}
|
1999-11-17 10:31:06 +08:00
|
|
|
|
1999-05-26 02:09:09 +08:00
|
|
|
/* Local functions */
|
|
|
|
|
2000-05-28 09:12:42 +08:00
|
|
|
extern void _initialize_d10v_tdep (void);
|
1999-05-26 02:09:09 +08:00
|
|
|
|
2000-05-28 09:12:42 +08:00
|
|
|
static void d10v_eva_prepare_to_trace (void);
|
1999-05-26 02:09:09 +08:00
|
|
|
|
2000-05-28 09:12:42 +08:00
|
|
|
static void d10v_eva_get_trace_data (void);
|
1999-04-16 09:35:26 +08:00
|
|
|
|
2000-04-21 12:21:21 +08:00
|
|
|
static CORE_ADDR
|
2003-05-01 23:37:45 +08:00
|
|
|
d10v_frame_align (struct gdbarch *gdbarch, CORE_ADDR sp)
|
2000-02-09 12:46:47 +08:00
|
|
|
{
|
2003-05-01 23:37:45 +08:00
|
|
|
/* Align to the size of an instruction (so that they can safely be
|
|
|
|
pushed onto the stack. */
|
|
|
|
return sp & ~3;
|
2000-02-09 12:46:47 +08:00
|
|
|
}
|
1999-04-16 09:35:26 +08:00
|
|
|
|
2002-04-19 02:09:09 +08:00
|
|
|
static const unsigned char *
|
2000-07-30 09:48:28 +08:00
|
|
|
d10v_breakpoint_from_pc (CORE_ADDR *pcptr, int *lenptr)
|
1999-05-26 02:09:09 +08:00
|
|
|
{
|
1999-07-08 04:19:36 +08:00
|
|
|
static unsigned char breakpoint[] =
|
|
|
|
{0x2f, 0x90, 0x5e, 0x00};
|
1999-05-26 02:09:09 +08:00
|
|
|
*lenptr = sizeof (breakpoint);
|
|
|
|
return breakpoint;
|
|
|
|
}
|
|
|
|
|
1999-11-17 10:31:06 +08:00
|
|
|
/* Map the REG_NR onto an ascii name. Return NULL or an empty string
|
2003-10-10 08:25:43 +08:00
|
|
|
when the reg_nr isn't valid. */
|
1999-11-17 10:31:06 +08:00
|
|
|
|
|
|
|
enum ts2_regnums
|
|
|
|
{
|
|
|
|
TS2_IMAP0_REGNUM = 32,
|
|
|
|
TS2_DMAP_REGNUM = 34,
|
|
|
|
TS2_NR_DMAP_REGS = 1,
|
|
|
|
TS2_A0_REGNUM = 35
|
|
|
|
};
|
|
|
|
|
2002-06-18 07:32:36 +08:00
|
|
|
static const char *
|
1999-11-17 10:31:06 +08:00
|
|
|
d10v_ts2_register_name (int reg_nr)
|
1999-05-26 02:09:09 +08:00
|
|
|
{
|
1999-07-08 04:19:36 +08:00
|
|
|
static char *register_names[] =
|
|
|
|
{
|
|
|
|
"r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7",
|
|
|
|
"r8", "r9", "r10", "r11", "r12", "r13", "r14", "r15",
|
|
|
|
"psw", "bpsw", "pc", "bpc", "cr4", "cr5", "cr6", "rpt_c",
|
|
|
|
"rpt_s", "rpt_e", "mod_s", "mod_e", "cr12", "cr13", "iba", "cr15",
|
|
|
|
"imap0", "imap1", "dmap", "a0", "a1"
|
1999-05-26 02:09:09 +08:00
|
|
|
};
|
|
|
|
if (reg_nr < 0)
|
|
|
|
return NULL;
|
|
|
|
if (reg_nr >= (sizeof (register_names) / sizeof (*register_names)))
|
|
|
|
return NULL;
|
1999-07-08 04:19:36 +08:00
|
|
|
return register_names[reg_nr];
|
1999-05-26 02:09:09 +08:00
|
|
|
}
|
|
|
|
|
1999-11-17 10:31:06 +08:00
|
|
|
enum ts3_regnums
|
|
|
|
{
|
|
|
|
TS3_IMAP0_REGNUM = 36,
|
|
|
|
TS3_DMAP0_REGNUM = 38,
|
|
|
|
TS3_NR_DMAP_REGS = 4,
|
|
|
|
TS3_A0_REGNUM = 32
|
|
|
|
};
|
|
|
|
|
2002-06-18 07:32:36 +08:00
|
|
|
static const char *
|
1999-11-17 10:31:06 +08:00
|
|
|
d10v_ts3_register_name (int reg_nr)
|
|
|
|
{
|
|
|
|
static char *register_names[] =
|
|
|
|
{
|
|
|
|
"r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7",
|
|
|
|
"r8", "r9", "r10", "r11", "r12", "r13", "r14", "r15",
|
|
|
|
"psw", "bpsw", "pc", "bpc", "cr4", "cr5", "cr6", "rpt_c",
|
|
|
|
"rpt_s", "rpt_e", "mod_s", "mod_e", "cr12", "cr13", "iba", "cr15",
|
|
|
|
"a0", "a1",
|
|
|
|
"spi", "spu",
|
|
|
|
"imap0", "imap1",
|
|
|
|
"dmap0", "dmap1", "dmap2", "dmap3"
|
|
|
|
};
|
|
|
|
if (reg_nr < 0)
|
|
|
|
return NULL;
|
|
|
|
if (reg_nr >= (sizeof (register_names) / sizeof (*register_names)))
|
|
|
|
return NULL;
|
|
|
|
return register_names[reg_nr];
|
|
|
|
}
|
|
|
|
|
2001-06-29 07:12:18 +08:00
|
|
|
/* Access the DMAP/IMAP registers in a target independent way.
|
|
|
|
|
|
|
|
Divide the D10V's 64k data space into four 16k segments:
|
|
|
|
0x0000 -- 0x3fff, 0x4000 -- 0x7fff, 0x8000 -- 0xbfff, and
|
|
|
|
0xc000 -- 0xffff.
|
|
|
|
|
|
|
|
On the TS2, the first two segments (0x0000 -- 0x3fff, 0x4000 --
|
|
|
|
0x7fff) always map to the on-chip data RAM, and the fourth always
|
|
|
|
maps to I/O space. The third (0x8000 - 0xbfff) can be mapped into
|
|
|
|
unified memory or instruction memory, under the control of the
|
|
|
|
single DMAP register.
|
|
|
|
|
|
|
|
On the TS3, there are four DMAP registers, each of which controls
|
|
|
|
one of the segments. */
|
1999-11-17 10:31:06 +08:00
|
|
|
|
|
|
|
static unsigned long
|
2003-05-08 03:21:13 +08:00
|
|
|
d10v_ts2_dmap_register (void *regcache, int reg_nr)
|
1999-11-17 10:31:06 +08:00
|
|
|
{
|
|
|
|
switch (reg_nr)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
case 1:
|
|
|
|
return 0x2000;
|
|
|
|
case 2:
|
2003-05-08 03:21:13 +08:00
|
|
|
{
|
|
|
|
ULONGEST reg;
|
|
|
|
regcache_cooked_read_unsigned (regcache, TS2_DMAP_REGNUM, ®);
|
|
|
|
return reg;
|
|
|
|
}
|
1999-11-17 10:31:06 +08:00
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned long
|
2003-05-08 03:21:13 +08:00
|
|
|
d10v_ts3_dmap_register (void *regcache, int reg_nr)
|
1999-11-17 10:31:06 +08:00
|
|
|
{
|
2003-05-08 03:21:13 +08:00
|
|
|
ULONGEST reg;
|
|
|
|
regcache_cooked_read_unsigned (regcache, TS3_DMAP0_REGNUM + reg_nr, ®);
|
|
|
|
return reg;
|
1999-11-17 10:31:06 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned long
|
2003-05-08 03:21:13 +08:00
|
|
|
d10v_ts2_imap_register (void *regcache, int reg_nr)
|
1999-11-17 10:31:06 +08:00
|
|
|
{
|
2003-05-08 03:21:13 +08:00
|
|
|
ULONGEST reg;
|
|
|
|
regcache_cooked_read_unsigned (regcache, TS2_IMAP0_REGNUM + reg_nr, ®);
|
|
|
|
return reg;
|
1999-11-17 10:31:06 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned long
|
2003-05-08 03:21:13 +08:00
|
|
|
d10v_ts3_imap_register (void *regcache, int reg_nr)
|
1999-11-17 10:31:06 +08:00
|
|
|
{
|
2003-05-08 03:21:13 +08:00
|
|
|
ULONGEST reg;
|
|
|
|
regcache_cooked_read_unsigned (regcache, TS3_IMAP0_REGNUM + reg_nr, ®);
|
|
|
|
return reg;
|
1999-11-17 10:31:06 +08:00
|
|
|
}
|
|
|
|
|
2003-09-17 Andrew Cagney <cagney@redhat.com>
* gdbarch.sh (DEPRECATED_REGISTER_BYTE): Rename REGISTER_BYTE.
* gdbarch.h, gdbarch.c: Regenerate.
* arm-linux-tdep.c, core-sol2.c, cris-tdep.c: Update.
* d10v-tdep.c, frame.c: Update.
* hppa-tdep.c, hppab-nat.c, hppah-nat.c, hppam3-nat.c: Update.
* hpux-thread.c, i386gnu-nat.c, ia64-aix-nat.c: Update.
* ia64-linux-nat.c, irix5-nat.c, lynx-nat.c, m68knbsd-nat.c: Update.
* mcore-tdep.c, mips-linux-tdep.c, mips-tdep.c: Update.
* mipsv4-nat.c, mn10300-tdep.c, ns32k-tdep.c: Update.
* ns32knbsd-nat.c, ppc-bdm.c, regcache.c, remote-sds.c: Update.
* remote-vx68.c, remote-vxmips.c, remote-vxsparc.c: Update.
* remote.c, rs6000-tdep.c, s390-tdep.c, sh64-tdep.c: Update.
* sparc-nat.c, sparc-tdep.c, sun3-nat.c, v850-tdep.c: Update.
* v850ice.c, vax-tdep.c, xstormy16-tdep.c: Update.
* config/m68k/tm-cisco.h, config/m68k/tm-delta68.h: Update.
* config/pa/nm-hppah.h: Update.
2003-09-17 Andrew Cagney <cagney@redhat.com>
* mi/mi-main.c: Rename REGISTER_BYTE to DEPRECATED_REGISTER_BYTE.
2003-09-17 22:24:31 +08:00
|
|
|
/* MAP GDB's internal register numbering (determined by the layout
|
|
|
|
from the DEPRECATED_REGISTER_BYTE array) onto the simulator's
|
2003-10-10 08:25:43 +08:00
|
|
|
register numbering. */
|
1999-11-17 10:31:06 +08:00
|
|
|
|
|
|
|
static int
|
|
|
|
d10v_ts2_register_sim_regno (int nr)
|
|
|
|
{
|
2003-03-29 09:40:01 +08:00
|
|
|
/* Only makes sense to supply raw registers. */
|
|
|
|
gdb_assert (nr >= 0 && nr < NUM_REGS);
|
1999-11-17 10:31:06 +08:00
|
|
|
if (nr >= TS2_IMAP0_REGNUM
|
|
|
|
&& nr < TS2_IMAP0_REGNUM + NR_IMAP_REGS)
|
|
|
|
return nr - TS2_IMAP0_REGNUM + SIM_D10V_IMAP0_REGNUM;
|
|
|
|
if (nr == TS2_DMAP_REGNUM)
|
|
|
|
return nr - TS2_DMAP_REGNUM + SIM_D10V_TS2_DMAP_REGNUM;
|
|
|
|
if (nr >= TS2_A0_REGNUM
|
|
|
|
&& nr < TS2_A0_REGNUM + NR_A_REGS)
|
|
|
|
return nr - TS2_A0_REGNUM + SIM_D10V_A0_REGNUM;
|
|
|
|
return nr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
d10v_ts3_register_sim_regno (int nr)
|
|
|
|
{
|
2003-03-29 09:40:01 +08:00
|
|
|
/* Only makes sense to supply raw registers. */
|
|
|
|
gdb_assert (nr >= 0 && nr < NUM_REGS);
|
1999-11-17 10:31:06 +08:00
|
|
|
if (nr >= TS3_IMAP0_REGNUM
|
|
|
|
&& nr < TS3_IMAP0_REGNUM + NR_IMAP_REGS)
|
|
|
|
return nr - TS3_IMAP0_REGNUM + SIM_D10V_IMAP0_REGNUM;
|
|
|
|
if (nr >= TS3_DMAP0_REGNUM
|
|
|
|
&& nr < TS3_DMAP0_REGNUM + TS3_NR_DMAP_REGS)
|
|
|
|
return nr - TS3_DMAP0_REGNUM + SIM_D10V_DMAP0_REGNUM;
|
|
|
|
if (nr >= TS3_A0_REGNUM
|
|
|
|
&& nr < TS3_A0_REGNUM + NR_A_REGS)
|
|
|
|
return nr - TS3_A0_REGNUM + SIM_D10V_A0_REGNUM;
|
|
|
|
return nr;
|
|
|
|
}
|
|
|
|
|
1999-05-26 02:09:09 +08:00
|
|
|
/* Return the GDB type object for the "standard" data type
|
|
|
|
of data in register N. */
|
|
|
|
|
* d10v-tdep.c (10v_frame_chain_valid, d10v_use_struct_convention,
d10v_breakpoint_from_pc, d10v_register_byte,
d10v_register_raw_size, d10v_register_virtual_size,
d10v_register_virtual_type, d10v_register_convertible,
d10v_register_convert_to_virtual, d10v_register_convert_to_raw,
d10v_make_daddr, d10v_make_iaddr, d10v_daddr_p, d10v_iaddr_p,
d10v_convert_iaddr_to_raw, d10v_convert_daddr_to_raw,
d10v_store_struct_return, d10v_store_return_value,
d10v_extract_struct_value_address, d10v_frame_saved_pc,
d10v_saved_pc_after_call, d10v_pop_frame, d10v_skip_prologue,
d10v_frame_chain, d10v_frame_init_saved_regs,
d10v_init_extra_frame_info, d10v_read_pc, d10v_write_pc,
d10v_read_sp, d10v_write_sp, d10v_write_fp, d10v_read_fp,
d10v_push_return_address, d10v_push_arguments,
d10v_extract_return_value): Make these functions static.
2001-06-29 06:10:41 +08:00
|
|
|
static struct type *
|
2003-03-02 01:59:12 +08:00
|
|
|
d10v_register_type (struct gdbarch *gdbarch, int reg_nr)
|
1999-05-26 02:09:09 +08:00
|
|
|
{
|
2003-05-03 07:56:12 +08:00
|
|
|
if (reg_nr == D10V_PC_REGNUM)
|
Clean up the D10V port so that GDB and the target program no
longer disagree on how big pointers are.
* findvar.c (value_from_register): Remove special case code for D10V.
* printcmd.c (print_frame_args): Same.
* valops.c (value_at, value_fetch_lazy): Same.
* values.c (unpack_long): Same.
* gdbarch.sh: Changes to effect the following:
* gdbarch.h (GDB_TARGET_IS_D10V, D10V_MAKE_DADDR,
gdbarch_d10v_make_daddr_ftype, gdbarch_d10v_make_daddr,
set_gdbarch_d10v_make_daddr, D10V_MAKE_IADDR,
gdbarch_d10v_make_iaddr_ftype, gdbarch_d10v_make_iaddr,
set_gdbarch_d10v_make_iaddr, D10V_DADDR_P,
gdbarch_d10v_daddr_p_ftype, gdbarch_d10v_daddr_p,
set_gdbarch_d10v_daddr_p, D10V_IADDR_P,
gdbarch_d10v_iaddr_p_ftype, gdbarch_d10v_iaddr_p,
set_gdbarch_d10v_iaddr_p, D10V_CONVERT_DADDR_TO_RAW,
gdbarch_d10v_convert_daddr_to_raw_ftype,
gdbarch_d10v_convert_daddr_to_raw,
set_gdbarch_d10v_convert_daddr_to_raw, D10V_CONVERT_IADDR_TO_RAW,
gdbarch_d10v_convert_iaddr_to_raw_ftype,
gdbarch_d10v_convert_iaddr_to_raw,
set_gdbarch_d10v_convert_iaddr_to_raw): Delete declarations.
* gdbarch.c: Delete the corresponding definitions.
(struct gdbarch): Delete members d10v_make_daddr,
d10v_make_iaddr, d10v_daddr_p, d10v_iaddr_p,
d10v_convert_daddr_to_raw, and d10v_convert_iaddr_to_raw.
(startup_gdbarch): Remove initializers for the above.
(verify_gdbarch, gdbarch_dump): Don't verify or dump them any
more.
* d10v-tdep.c (d10v_register_virtual_type): Rather that
claiming the stack pointer and PC are 32 bits long (which they
aren't), say that the stack pointer is an int16_t, and the
program counter is a function pointer. This allows the rest
of GDB to make the appropriate conversions between the code
pointer format and real addresses.
(d10v_register_convertible, d10v_register_convert_to_virtual,
d10v_register_convert_to_raw): Delete function; no registers
are convertible now, so we use
generic_register_convertible_not instead.
(d10v_address_to_pointer, d10v_pointer_to_address): New gdbarch
methods.
(d10v_push_arguments, d10v_extract_return_value): Remove special
cases for code and data pointers.
(d10v_gdbarch_init): Set gdbarch_ptr_bit to 16, so that GDB and
the target agree on how large pointers are. Say that addresses
are 32 bits long. Register the address_to_pointer and
pointer_to_address conversion functions. Since no registers are
convertible now, register generic_register_convertible_not as the
gdbarch_register_convertible method instead of
d10v_register_convertible. Remove registrations for
d10v_register_convert_to_virtual,
d10v_register_convert_to_raw, gdbarch_d10v_make_daddr,
gdbarch_d10v_make_iaddr, gdbarch_d10v_daddr_p,
gdbarch_d10v_iaddr_p, gdbarch_d10v_convert_daddr_to_raw, and
gdbarch_d10v_convert_iaddr_to_raw.
2001-07-11 05:24:48 +08:00
|
|
|
return builtin_type_void_func_ptr;
|
2003-04-26 05:20:58 +08:00
|
|
|
if (reg_nr == D10V_SP_REGNUM || reg_nr == D10V_FP_REGNUM)
|
2002-06-07 00:35:37 +08:00
|
|
|
return builtin_type_void_data_ptr;
|
2003-04-26 05:20:58 +08:00
|
|
|
else if (reg_nr >= a0_regnum (gdbarch)
|
|
|
|
&& reg_nr < (a0_regnum (gdbarch) + NR_A_REGS))
|
1999-11-17 10:31:06 +08:00
|
|
|
return builtin_type_int64;
|
1999-05-26 02:09:09 +08:00
|
|
|
else
|
1999-11-17 10:31:06 +08:00
|
|
|
return builtin_type_int16;
|
1999-05-26 02:09:09 +08:00
|
|
|
}
|
|
|
|
|
* d10v-tdep.c (10v_frame_chain_valid, d10v_use_struct_convention,
d10v_breakpoint_from_pc, d10v_register_byte,
d10v_register_raw_size, d10v_register_virtual_size,
d10v_register_virtual_type, d10v_register_convertible,
d10v_register_convert_to_virtual, d10v_register_convert_to_raw,
d10v_make_daddr, d10v_make_iaddr, d10v_daddr_p, d10v_iaddr_p,
d10v_convert_iaddr_to_raw, d10v_convert_daddr_to_raw,
d10v_store_struct_return, d10v_store_return_value,
d10v_extract_struct_value_address, d10v_frame_saved_pc,
d10v_saved_pc_after_call, d10v_pop_frame, d10v_skip_prologue,
d10v_frame_chain, d10v_frame_init_saved_regs,
d10v_init_extra_frame_info, d10v_read_pc, d10v_write_pc,
d10v_read_sp, d10v_write_sp, d10v_write_fp, d10v_read_fp,
d10v_push_return_address, d10v_push_arguments,
d10v_extract_return_value): Make these functions static.
2001-06-29 06:10:41 +08:00
|
|
|
static int
|
2000-07-30 09:48:28 +08:00
|
|
|
d10v_daddr_p (CORE_ADDR x)
|
1999-05-26 02:09:09 +08:00
|
|
|
{
|
|
|
|
return (((x) & 0x3000000) == DMEM_START);
|
|
|
|
}
|
|
|
|
|
* d10v-tdep.c (10v_frame_chain_valid, d10v_use_struct_convention,
d10v_breakpoint_from_pc, d10v_register_byte,
d10v_register_raw_size, d10v_register_virtual_size,
d10v_register_virtual_type, d10v_register_convertible,
d10v_register_convert_to_virtual, d10v_register_convert_to_raw,
d10v_make_daddr, d10v_make_iaddr, d10v_daddr_p, d10v_iaddr_p,
d10v_convert_iaddr_to_raw, d10v_convert_daddr_to_raw,
d10v_store_struct_return, d10v_store_return_value,
d10v_extract_struct_value_address, d10v_frame_saved_pc,
d10v_saved_pc_after_call, d10v_pop_frame, d10v_skip_prologue,
d10v_frame_chain, d10v_frame_init_saved_regs,
d10v_init_extra_frame_info, d10v_read_pc, d10v_write_pc,
d10v_read_sp, d10v_write_sp, d10v_write_fp, d10v_read_fp,
d10v_push_return_address, d10v_push_arguments,
d10v_extract_return_value): Make these functions static.
2001-06-29 06:10:41 +08:00
|
|
|
static int
|
2000-07-30 09:48:28 +08:00
|
|
|
d10v_iaddr_p (CORE_ADDR x)
|
1999-05-26 02:09:09 +08:00
|
|
|
{
|
|
|
|
return (((x) & 0x3000000) == IMEM_START);
|
|
|
|
}
|
|
|
|
|
2002-04-13 06:50:42 +08:00
|
|
|
static CORE_ADDR
|
|
|
|
d10v_make_daddr (CORE_ADDR x)
|
|
|
|
{
|
|
|
|
return ((x) | DMEM_START);
|
|
|
|
}
|
|
|
|
|
|
|
|
static CORE_ADDR
|
|
|
|
d10v_make_iaddr (CORE_ADDR x)
|
|
|
|
{
|
|
|
|
if (d10v_iaddr_p (x))
|
2003-10-10 08:25:43 +08:00
|
|
|
return x; /* Idempotency -- x is already in the IMEM space. */
|
2002-04-13 06:50:42 +08:00
|
|
|
else
|
|
|
|
return (((x) << 2) | IMEM_START);
|
|
|
|
}
|
1999-05-26 02:09:09 +08:00
|
|
|
|
* d10v-tdep.c (10v_frame_chain_valid, d10v_use_struct_convention,
d10v_breakpoint_from_pc, d10v_register_byte,
d10v_register_raw_size, d10v_register_virtual_size,
d10v_register_virtual_type, d10v_register_convertible,
d10v_register_convert_to_virtual, d10v_register_convert_to_raw,
d10v_make_daddr, d10v_make_iaddr, d10v_daddr_p, d10v_iaddr_p,
d10v_convert_iaddr_to_raw, d10v_convert_daddr_to_raw,
d10v_store_struct_return, d10v_store_return_value,
d10v_extract_struct_value_address, d10v_frame_saved_pc,
d10v_saved_pc_after_call, d10v_pop_frame, d10v_skip_prologue,
d10v_frame_chain, d10v_frame_init_saved_regs,
d10v_init_extra_frame_info, d10v_read_pc, d10v_write_pc,
d10v_read_sp, d10v_write_sp, d10v_write_fp, d10v_read_fp,
d10v_push_return_address, d10v_push_arguments,
d10v_extract_return_value): Make these functions static.
2001-06-29 06:10:41 +08:00
|
|
|
static CORE_ADDR
|
2000-07-30 09:48:28 +08:00
|
|
|
d10v_convert_iaddr_to_raw (CORE_ADDR x)
|
1999-05-26 02:09:09 +08:00
|
|
|
{
|
|
|
|
return (((x) >> 2) & 0xffff);
|
|
|
|
}
|
|
|
|
|
* d10v-tdep.c (10v_frame_chain_valid, d10v_use_struct_convention,
d10v_breakpoint_from_pc, d10v_register_byte,
d10v_register_raw_size, d10v_register_virtual_size,
d10v_register_virtual_type, d10v_register_convertible,
d10v_register_convert_to_virtual, d10v_register_convert_to_raw,
d10v_make_daddr, d10v_make_iaddr, d10v_daddr_p, d10v_iaddr_p,
d10v_convert_iaddr_to_raw, d10v_convert_daddr_to_raw,
d10v_store_struct_return, d10v_store_return_value,
d10v_extract_struct_value_address, d10v_frame_saved_pc,
d10v_saved_pc_after_call, d10v_pop_frame, d10v_skip_prologue,
d10v_frame_chain, d10v_frame_init_saved_regs,
d10v_init_extra_frame_info, d10v_read_pc, d10v_write_pc,
d10v_read_sp, d10v_write_sp, d10v_write_fp, d10v_read_fp,
d10v_push_return_address, d10v_push_arguments,
d10v_extract_return_value): Make these functions static.
2001-06-29 06:10:41 +08:00
|
|
|
static CORE_ADDR
|
2000-07-30 09:48:28 +08:00
|
|
|
d10v_convert_daddr_to_raw (CORE_ADDR x)
|
1999-05-26 02:09:09 +08:00
|
|
|
{
|
|
|
|
return ((x) & 0xffff);
|
|
|
|
}
|
|
|
|
|
Clean up the D10V port so that GDB and the target program no
longer disagree on how big pointers are.
* findvar.c (value_from_register): Remove special case code for D10V.
* printcmd.c (print_frame_args): Same.
* valops.c (value_at, value_fetch_lazy): Same.
* values.c (unpack_long): Same.
* gdbarch.sh: Changes to effect the following:
* gdbarch.h (GDB_TARGET_IS_D10V, D10V_MAKE_DADDR,
gdbarch_d10v_make_daddr_ftype, gdbarch_d10v_make_daddr,
set_gdbarch_d10v_make_daddr, D10V_MAKE_IADDR,
gdbarch_d10v_make_iaddr_ftype, gdbarch_d10v_make_iaddr,
set_gdbarch_d10v_make_iaddr, D10V_DADDR_P,
gdbarch_d10v_daddr_p_ftype, gdbarch_d10v_daddr_p,
set_gdbarch_d10v_daddr_p, D10V_IADDR_P,
gdbarch_d10v_iaddr_p_ftype, gdbarch_d10v_iaddr_p,
set_gdbarch_d10v_iaddr_p, D10V_CONVERT_DADDR_TO_RAW,
gdbarch_d10v_convert_daddr_to_raw_ftype,
gdbarch_d10v_convert_daddr_to_raw,
set_gdbarch_d10v_convert_daddr_to_raw, D10V_CONVERT_IADDR_TO_RAW,
gdbarch_d10v_convert_iaddr_to_raw_ftype,
gdbarch_d10v_convert_iaddr_to_raw,
set_gdbarch_d10v_convert_iaddr_to_raw): Delete declarations.
* gdbarch.c: Delete the corresponding definitions.
(struct gdbarch): Delete members d10v_make_daddr,
d10v_make_iaddr, d10v_daddr_p, d10v_iaddr_p,
d10v_convert_daddr_to_raw, and d10v_convert_iaddr_to_raw.
(startup_gdbarch): Remove initializers for the above.
(verify_gdbarch, gdbarch_dump): Don't verify or dump them any
more.
* d10v-tdep.c (d10v_register_virtual_type): Rather that
claiming the stack pointer and PC are 32 bits long (which they
aren't), say that the stack pointer is an int16_t, and the
program counter is a function pointer. This allows the rest
of GDB to make the appropriate conversions between the code
pointer format and real addresses.
(d10v_register_convertible, d10v_register_convert_to_virtual,
d10v_register_convert_to_raw): Delete function; no registers
are convertible now, so we use
generic_register_convertible_not instead.
(d10v_address_to_pointer, d10v_pointer_to_address): New gdbarch
methods.
(d10v_push_arguments, d10v_extract_return_value): Remove special
cases for code and data pointers.
(d10v_gdbarch_init): Set gdbarch_ptr_bit to 16, so that GDB and
the target agree on how large pointers are. Say that addresses
are 32 bits long. Register the address_to_pointer and
pointer_to_address conversion functions. Since no registers are
convertible now, register generic_register_convertible_not as the
gdbarch_register_convertible method instead of
d10v_register_convertible. Remove registrations for
d10v_register_convert_to_virtual,
d10v_register_convert_to_raw, gdbarch_d10v_make_daddr,
gdbarch_d10v_make_iaddr, gdbarch_d10v_daddr_p,
gdbarch_d10v_iaddr_p, gdbarch_d10v_convert_daddr_to_raw, and
gdbarch_d10v_convert_iaddr_to_raw.
2001-07-11 05:24:48 +08:00
|
|
|
static void
|
|
|
|
d10v_address_to_pointer (struct type *type, void *buf, CORE_ADDR addr)
|
|
|
|
{
|
|
|
|
/* Is it a code address? */
|
|
|
|
if (TYPE_CODE (TYPE_TARGET_TYPE (type)) == TYPE_CODE_FUNC
|
|
|
|
|| TYPE_CODE (TYPE_TARGET_TYPE (type)) == TYPE_CODE_METHOD)
|
|
|
|
{
|
|
|
|
store_unsigned_integer (buf, TYPE_LENGTH (type),
|
|
|
|
d10v_convert_iaddr_to_raw (addr));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Strip off any upper segment bits. */
|
|
|
|
store_unsigned_integer (buf, TYPE_LENGTH (type),
|
|
|
|
d10v_convert_daddr_to_raw (addr));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static CORE_ADDR
|
2003-01-07 02:49:09 +08:00
|
|
|
d10v_pointer_to_address (struct type *type, const void *buf)
|
Clean up the D10V port so that GDB and the target program no
longer disagree on how big pointers are.
* findvar.c (value_from_register): Remove special case code for D10V.
* printcmd.c (print_frame_args): Same.
* valops.c (value_at, value_fetch_lazy): Same.
* values.c (unpack_long): Same.
* gdbarch.sh: Changes to effect the following:
* gdbarch.h (GDB_TARGET_IS_D10V, D10V_MAKE_DADDR,
gdbarch_d10v_make_daddr_ftype, gdbarch_d10v_make_daddr,
set_gdbarch_d10v_make_daddr, D10V_MAKE_IADDR,
gdbarch_d10v_make_iaddr_ftype, gdbarch_d10v_make_iaddr,
set_gdbarch_d10v_make_iaddr, D10V_DADDR_P,
gdbarch_d10v_daddr_p_ftype, gdbarch_d10v_daddr_p,
set_gdbarch_d10v_daddr_p, D10V_IADDR_P,
gdbarch_d10v_iaddr_p_ftype, gdbarch_d10v_iaddr_p,
set_gdbarch_d10v_iaddr_p, D10V_CONVERT_DADDR_TO_RAW,
gdbarch_d10v_convert_daddr_to_raw_ftype,
gdbarch_d10v_convert_daddr_to_raw,
set_gdbarch_d10v_convert_daddr_to_raw, D10V_CONVERT_IADDR_TO_RAW,
gdbarch_d10v_convert_iaddr_to_raw_ftype,
gdbarch_d10v_convert_iaddr_to_raw,
set_gdbarch_d10v_convert_iaddr_to_raw): Delete declarations.
* gdbarch.c: Delete the corresponding definitions.
(struct gdbarch): Delete members d10v_make_daddr,
d10v_make_iaddr, d10v_daddr_p, d10v_iaddr_p,
d10v_convert_daddr_to_raw, and d10v_convert_iaddr_to_raw.
(startup_gdbarch): Remove initializers for the above.
(verify_gdbarch, gdbarch_dump): Don't verify or dump them any
more.
* d10v-tdep.c (d10v_register_virtual_type): Rather that
claiming the stack pointer and PC are 32 bits long (which they
aren't), say that the stack pointer is an int16_t, and the
program counter is a function pointer. This allows the rest
of GDB to make the appropriate conversions between the code
pointer format and real addresses.
(d10v_register_convertible, d10v_register_convert_to_virtual,
d10v_register_convert_to_raw): Delete function; no registers
are convertible now, so we use
generic_register_convertible_not instead.
(d10v_address_to_pointer, d10v_pointer_to_address): New gdbarch
methods.
(d10v_push_arguments, d10v_extract_return_value): Remove special
cases for code and data pointers.
(d10v_gdbarch_init): Set gdbarch_ptr_bit to 16, so that GDB and
the target agree on how large pointers are. Say that addresses
are 32 bits long. Register the address_to_pointer and
pointer_to_address conversion functions. Since no registers are
convertible now, register generic_register_convertible_not as the
gdbarch_register_convertible method instead of
d10v_register_convertible. Remove registrations for
d10v_register_convert_to_virtual,
d10v_register_convert_to_raw, gdbarch_d10v_make_daddr,
gdbarch_d10v_make_iaddr, gdbarch_d10v_daddr_p,
gdbarch_d10v_iaddr_p, gdbarch_d10v_convert_daddr_to_raw, and
gdbarch_d10v_convert_iaddr_to_raw.
2001-07-11 05:24:48 +08:00
|
|
|
{
|
2003-04-26 05:20:58 +08:00
|
|
|
CORE_ADDR addr = extract_unsigned_integer (buf, TYPE_LENGTH (type));
|
Clean up the D10V port so that GDB and the target program no
longer disagree on how big pointers are.
* findvar.c (value_from_register): Remove special case code for D10V.
* printcmd.c (print_frame_args): Same.
* valops.c (value_at, value_fetch_lazy): Same.
* values.c (unpack_long): Same.
* gdbarch.sh: Changes to effect the following:
* gdbarch.h (GDB_TARGET_IS_D10V, D10V_MAKE_DADDR,
gdbarch_d10v_make_daddr_ftype, gdbarch_d10v_make_daddr,
set_gdbarch_d10v_make_daddr, D10V_MAKE_IADDR,
gdbarch_d10v_make_iaddr_ftype, gdbarch_d10v_make_iaddr,
set_gdbarch_d10v_make_iaddr, D10V_DADDR_P,
gdbarch_d10v_daddr_p_ftype, gdbarch_d10v_daddr_p,
set_gdbarch_d10v_daddr_p, D10V_IADDR_P,
gdbarch_d10v_iaddr_p_ftype, gdbarch_d10v_iaddr_p,
set_gdbarch_d10v_iaddr_p, D10V_CONVERT_DADDR_TO_RAW,
gdbarch_d10v_convert_daddr_to_raw_ftype,
gdbarch_d10v_convert_daddr_to_raw,
set_gdbarch_d10v_convert_daddr_to_raw, D10V_CONVERT_IADDR_TO_RAW,
gdbarch_d10v_convert_iaddr_to_raw_ftype,
gdbarch_d10v_convert_iaddr_to_raw,
set_gdbarch_d10v_convert_iaddr_to_raw): Delete declarations.
* gdbarch.c: Delete the corresponding definitions.
(struct gdbarch): Delete members d10v_make_daddr,
d10v_make_iaddr, d10v_daddr_p, d10v_iaddr_p,
d10v_convert_daddr_to_raw, and d10v_convert_iaddr_to_raw.
(startup_gdbarch): Remove initializers for the above.
(verify_gdbarch, gdbarch_dump): Don't verify or dump them any
more.
* d10v-tdep.c (d10v_register_virtual_type): Rather that
claiming the stack pointer and PC are 32 bits long (which they
aren't), say that the stack pointer is an int16_t, and the
program counter is a function pointer. This allows the rest
of GDB to make the appropriate conversions between the code
pointer format and real addresses.
(d10v_register_convertible, d10v_register_convert_to_virtual,
d10v_register_convert_to_raw): Delete function; no registers
are convertible now, so we use
generic_register_convertible_not instead.
(d10v_address_to_pointer, d10v_pointer_to_address): New gdbarch
methods.
(d10v_push_arguments, d10v_extract_return_value): Remove special
cases for code and data pointers.
(d10v_gdbarch_init): Set gdbarch_ptr_bit to 16, so that GDB and
the target agree on how large pointers are. Say that addresses
are 32 bits long. Register the address_to_pointer and
pointer_to_address conversion functions. Since no registers are
convertible now, register generic_register_convertible_not as the
gdbarch_register_convertible method instead of
d10v_register_convertible. Remove registrations for
d10v_register_convert_to_virtual,
d10v_register_convert_to_raw, gdbarch_d10v_make_daddr,
gdbarch_d10v_make_iaddr, gdbarch_d10v_daddr_p,
gdbarch_d10v_iaddr_p, gdbarch_d10v_convert_daddr_to_raw, and
gdbarch_d10v_convert_iaddr_to_raw.
2001-07-11 05:24:48 +08:00
|
|
|
/* Is it a code address? */
|
|
|
|
if (TYPE_CODE (TYPE_TARGET_TYPE (type)) == TYPE_CODE_FUNC
|
2001-12-12 10:11:52 +08:00
|
|
|
|| TYPE_CODE (TYPE_TARGET_TYPE (type)) == TYPE_CODE_METHOD
|
|
|
|
|| TYPE_CODE_SPACE (TYPE_TARGET_TYPE (type)))
|
Clean up the D10V port so that GDB and the target program no
longer disagree on how big pointers are.
* findvar.c (value_from_register): Remove special case code for D10V.
* printcmd.c (print_frame_args): Same.
* valops.c (value_at, value_fetch_lazy): Same.
* values.c (unpack_long): Same.
* gdbarch.sh: Changes to effect the following:
* gdbarch.h (GDB_TARGET_IS_D10V, D10V_MAKE_DADDR,
gdbarch_d10v_make_daddr_ftype, gdbarch_d10v_make_daddr,
set_gdbarch_d10v_make_daddr, D10V_MAKE_IADDR,
gdbarch_d10v_make_iaddr_ftype, gdbarch_d10v_make_iaddr,
set_gdbarch_d10v_make_iaddr, D10V_DADDR_P,
gdbarch_d10v_daddr_p_ftype, gdbarch_d10v_daddr_p,
set_gdbarch_d10v_daddr_p, D10V_IADDR_P,
gdbarch_d10v_iaddr_p_ftype, gdbarch_d10v_iaddr_p,
set_gdbarch_d10v_iaddr_p, D10V_CONVERT_DADDR_TO_RAW,
gdbarch_d10v_convert_daddr_to_raw_ftype,
gdbarch_d10v_convert_daddr_to_raw,
set_gdbarch_d10v_convert_daddr_to_raw, D10V_CONVERT_IADDR_TO_RAW,
gdbarch_d10v_convert_iaddr_to_raw_ftype,
gdbarch_d10v_convert_iaddr_to_raw,
set_gdbarch_d10v_convert_iaddr_to_raw): Delete declarations.
* gdbarch.c: Delete the corresponding definitions.
(struct gdbarch): Delete members d10v_make_daddr,
d10v_make_iaddr, d10v_daddr_p, d10v_iaddr_p,
d10v_convert_daddr_to_raw, and d10v_convert_iaddr_to_raw.
(startup_gdbarch): Remove initializers for the above.
(verify_gdbarch, gdbarch_dump): Don't verify or dump them any
more.
* d10v-tdep.c (d10v_register_virtual_type): Rather that
claiming the stack pointer and PC are 32 bits long (which they
aren't), say that the stack pointer is an int16_t, and the
program counter is a function pointer. This allows the rest
of GDB to make the appropriate conversions between the code
pointer format and real addresses.
(d10v_register_convertible, d10v_register_convert_to_virtual,
d10v_register_convert_to_raw): Delete function; no registers
are convertible now, so we use
generic_register_convertible_not instead.
(d10v_address_to_pointer, d10v_pointer_to_address): New gdbarch
methods.
(d10v_push_arguments, d10v_extract_return_value): Remove special
cases for code and data pointers.
(d10v_gdbarch_init): Set gdbarch_ptr_bit to 16, so that GDB and
the target agree on how large pointers are. Say that addresses
are 32 bits long. Register the address_to_pointer and
pointer_to_address conversion functions. Since no registers are
convertible now, register generic_register_convertible_not as the
gdbarch_register_convertible method instead of
d10v_register_convertible. Remove registrations for
d10v_register_convert_to_virtual,
d10v_register_convert_to_raw, gdbarch_d10v_make_daddr,
gdbarch_d10v_make_iaddr, gdbarch_d10v_daddr_p,
gdbarch_d10v_iaddr_p, gdbarch_d10v_convert_daddr_to_raw, and
gdbarch_d10v_convert_iaddr_to_raw.
2001-07-11 05:24:48 +08:00
|
|
|
return d10v_make_iaddr (addr);
|
|
|
|
else
|
|
|
|
return d10v_make_daddr (addr);
|
|
|
|
}
|
|
|
|
|
2002-06-07 00:35:37 +08:00
|
|
|
/* Don't do anything if we have an integer, this way users can type 'x
|
|
|
|
<addr>' w/o having gdb outsmart them. The internal gdb conversions
|
|
|
|
to the correct space are taken care of in the pointer_to_address
|
|
|
|
function. If we don't do this, 'x $fp' wouldn't work. */
|
2001-10-16 02:18:30 +08:00
|
|
|
static CORE_ADDR
|
|
|
|
d10v_integer_to_address (struct type *type, void *buf)
|
|
|
|
{
|
|
|
|
LONGEST val;
|
|
|
|
val = unpack_long (type, buf);
|
2002-06-07 00:35:37 +08:00
|
|
|
return val;
|
2001-10-16 02:18:30 +08:00
|
|
|
}
|
Clean up the D10V port so that GDB and the target program no
longer disagree on how big pointers are.
* findvar.c (value_from_register): Remove special case code for D10V.
* printcmd.c (print_frame_args): Same.
* valops.c (value_at, value_fetch_lazy): Same.
* values.c (unpack_long): Same.
* gdbarch.sh: Changes to effect the following:
* gdbarch.h (GDB_TARGET_IS_D10V, D10V_MAKE_DADDR,
gdbarch_d10v_make_daddr_ftype, gdbarch_d10v_make_daddr,
set_gdbarch_d10v_make_daddr, D10V_MAKE_IADDR,
gdbarch_d10v_make_iaddr_ftype, gdbarch_d10v_make_iaddr,
set_gdbarch_d10v_make_iaddr, D10V_DADDR_P,
gdbarch_d10v_daddr_p_ftype, gdbarch_d10v_daddr_p,
set_gdbarch_d10v_daddr_p, D10V_IADDR_P,
gdbarch_d10v_iaddr_p_ftype, gdbarch_d10v_iaddr_p,
set_gdbarch_d10v_iaddr_p, D10V_CONVERT_DADDR_TO_RAW,
gdbarch_d10v_convert_daddr_to_raw_ftype,
gdbarch_d10v_convert_daddr_to_raw,
set_gdbarch_d10v_convert_daddr_to_raw, D10V_CONVERT_IADDR_TO_RAW,
gdbarch_d10v_convert_iaddr_to_raw_ftype,
gdbarch_d10v_convert_iaddr_to_raw,
set_gdbarch_d10v_convert_iaddr_to_raw): Delete declarations.
* gdbarch.c: Delete the corresponding definitions.
(struct gdbarch): Delete members d10v_make_daddr,
d10v_make_iaddr, d10v_daddr_p, d10v_iaddr_p,
d10v_convert_daddr_to_raw, and d10v_convert_iaddr_to_raw.
(startup_gdbarch): Remove initializers for the above.
(verify_gdbarch, gdbarch_dump): Don't verify or dump them any
more.
* d10v-tdep.c (d10v_register_virtual_type): Rather that
claiming the stack pointer and PC are 32 bits long (which they
aren't), say that the stack pointer is an int16_t, and the
program counter is a function pointer. This allows the rest
of GDB to make the appropriate conversions between the code
pointer format and real addresses.
(d10v_register_convertible, d10v_register_convert_to_virtual,
d10v_register_convert_to_raw): Delete function; no registers
are convertible now, so we use
generic_register_convertible_not instead.
(d10v_address_to_pointer, d10v_pointer_to_address): New gdbarch
methods.
(d10v_push_arguments, d10v_extract_return_value): Remove special
cases for code and data pointers.
(d10v_gdbarch_init): Set gdbarch_ptr_bit to 16, so that GDB and
the target agree on how large pointers are. Say that addresses
are 32 bits long. Register the address_to_pointer and
pointer_to_address conversion functions. Since no registers are
convertible now, register generic_register_convertible_not as the
gdbarch_register_convertible method instead of
d10v_register_convertible. Remove registrations for
d10v_register_convert_to_virtual,
d10v_register_convert_to_raw, gdbarch_d10v_make_daddr,
gdbarch_d10v_make_iaddr, gdbarch_d10v_daddr_p,
gdbarch_d10v_iaddr_p, gdbarch_d10v_convert_daddr_to_raw, and
gdbarch_d10v_convert_iaddr_to_raw.
2001-07-11 05:24:48 +08:00
|
|
|
|
2003-11-09 22:20:55 +08:00
|
|
|
/* Handle the d10v's return_value convention. */
|
1999-05-26 02:09:09 +08:00
|
|
|
|
2003-11-09 22:20:55 +08:00
|
|
|
static enum return_value_convention
|
|
|
|
d10v_return_value (struct gdbarch *gdbarch, struct type *valtype,
|
2003-11-11 06:47:31 +08:00
|
|
|
struct regcache *regcache, void *readbuf,
|
|
|
|
const void *writebuf)
|
1999-05-26 02:09:09 +08:00
|
|
|
{
|
2003-11-09 22:20:55 +08:00
|
|
|
if (TYPE_LENGTH (valtype) > 8)
|
|
|
|
/* Anything larger than 8 bytes (4 registers) goes on the stack. */
|
|
|
|
return RETURN_VALUE_STRUCT_CONVENTION;
|
|
|
|
if (TYPE_LENGTH (valtype) == 5
|
|
|
|
|| TYPE_LENGTH (valtype) == 6)
|
|
|
|
/* Anything 5 or 6 bytes in size goes in memory. Contents don't
|
|
|
|
appear to matter. Note that 7 and 8 byte objects do end up in
|
|
|
|
registers! */
|
|
|
|
return RETURN_VALUE_STRUCT_CONVENTION;
|
|
|
|
if (TYPE_LENGTH (valtype) == 1)
|
2002-06-07 02:47:35 +08:00
|
|
|
{
|
2003-11-09 22:20:55 +08:00
|
|
|
/* All single byte values go in a register stored right-aligned.
|
|
|
|
Note: 2 byte integer values are handled further down. */
|
|
|
|
if (readbuf)
|
|
|
|
{
|
|
|
|
/* Since TYPE is smaller than the register, there isn't a
|
|
|
|
sign extension problem. Let the extraction truncate the
|
|
|
|
register value. */
|
|
|
|
ULONGEST regval;
|
|
|
|
regcache_cooked_read_unsigned (regcache, R0_REGNUM,
|
|
|
|
®val);
|
|
|
|
store_unsigned_integer (readbuf, TYPE_LENGTH (valtype), regval);
|
|
|
|
|
|
|
|
}
|
|
|
|
if (writebuf)
|
|
|
|
{
|
|
|
|
ULONGEST regval;
|
|
|
|
if (TYPE_CODE (valtype) == TYPE_CODE_INT)
|
|
|
|
/* Some sort of integer value stored in R0. Use
|
|
|
|
unpack_long since that should handle any required sign
|
|
|
|
extension. */
|
|
|
|
regval = unpack_long (valtype, writebuf);
|
|
|
|
else
|
|
|
|
/* Some other type. Don't sign-extend the value when
|
|
|
|
storing it in the register. */
|
|
|
|
regval = extract_unsigned_integer (writebuf, 1);
|
|
|
|
regcache_cooked_write_unsigned (regcache, R0_REGNUM, regval);
|
|
|
|
}
|
|
|
|
return RETURN_VALUE_REGISTER_CONVENTION;
|
2002-06-07 02:47:35 +08:00
|
|
|
}
|
2003-11-09 22:20:55 +08:00
|
|
|
if ((TYPE_CODE (valtype) == TYPE_CODE_STRUCT
|
|
|
|
|| TYPE_CODE (valtype) == TYPE_CODE_UNION)
|
|
|
|
&& TYPE_NFIELDS (valtype) > 1
|
|
|
|
&& TYPE_FIELD_BITPOS (valtype, 1) == 8)
|
|
|
|
/* If a composite is 8 bit aligned (determined by looking at the
|
|
|
|
start address of the second field), put it in memory. */
|
|
|
|
return RETURN_VALUE_STRUCT_CONVENTION;
|
|
|
|
/* Assume it is in registers. */
|
|
|
|
if (writebuf || readbuf)
|
2003-01-13 02:59:53 +08:00
|
|
|
{
|
|
|
|
int reg;
|
2003-11-09 22:20:55 +08:00
|
|
|
/* Per above, the value is never more than 8 bytes long. */
|
|
|
|
gdb_assert (TYPE_LENGTH (valtype) <= 8);
|
|
|
|
/* Xfer 2 bytes at a time. */
|
|
|
|
for (reg = 0; (reg * 2) + 1 < TYPE_LENGTH (valtype); reg++)
|
|
|
|
{
|
|
|
|
if (readbuf)
|
|
|
|
regcache_cooked_read (regcache, R0_REGNUM + reg,
|
|
|
|
(bfd_byte *) readbuf + reg * 2);
|
|
|
|
if (writebuf)
|
|
|
|
regcache_cooked_write (regcache, R0_REGNUM + reg,
|
|
|
|
(bfd_byte *) writebuf + reg * 2);
|
|
|
|
}
|
|
|
|
/* Any trailing byte ends up _left_ aligned. */
|
|
|
|
if ((reg * 2) < TYPE_LENGTH (valtype))
|
2003-01-13 02:59:53 +08:00
|
|
|
{
|
2003-11-09 22:20:55 +08:00
|
|
|
if (readbuf)
|
|
|
|
regcache_cooked_read_part (regcache, R0_REGNUM + reg,
|
|
|
|
0, 1, (bfd_byte *) readbuf + reg * 2);
|
|
|
|
if (writebuf)
|
|
|
|
regcache_cooked_write_part (regcache, R0_REGNUM + reg,
|
|
|
|
0, 1, (bfd_byte *) writebuf + reg * 2);
|
2003-01-13 02:59:53 +08:00
|
|
|
}
|
|
|
|
}
|
2003-11-09 22:20:55 +08:00
|
|
|
return RETURN_VALUE_REGISTER_CONVENTION;
|
1999-05-26 02:09:09 +08:00
|
|
|
}
|
|
|
|
|
1999-07-08 04:19:36 +08:00
|
|
|
static int
|
2000-07-30 09:48:28 +08:00
|
|
|
check_prologue (unsigned short op)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
|
|
|
/* st rn, @-sp */
|
|
|
|
if ((op & 0x7E1F) == 0x6C1F)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
/* st2w rn, @-sp */
|
|
|
|
if ((op & 0x7E3F) == 0x6E1F)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
/* subi sp, n */
|
|
|
|
if ((op & 0x7FE1) == 0x01E1)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
/* mv r11, sp */
|
|
|
|
if (op == 0x417E)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
/* nop */
|
|
|
|
if (op == 0x5E00)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
/* st rn, @sp */
|
|
|
|
if ((op & 0x7E1F) == 0x681E)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
/* st2w rn, @sp */
|
1999-07-08 04:19:36 +08:00
|
|
|
if ((op & 0x7E3F) == 0x3A1E)
|
|
|
|
return 1;
|
1999-04-16 09:35:26 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
* d10v-tdep.c (10v_frame_chain_valid, d10v_use_struct_convention,
d10v_breakpoint_from_pc, d10v_register_byte,
d10v_register_raw_size, d10v_register_virtual_size,
d10v_register_virtual_type, d10v_register_convertible,
d10v_register_convert_to_virtual, d10v_register_convert_to_raw,
d10v_make_daddr, d10v_make_iaddr, d10v_daddr_p, d10v_iaddr_p,
d10v_convert_iaddr_to_raw, d10v_convert_daddr_to_raw,
d10v_store_struct_return, d10v_store_return_value,
d10v_extract_struct_value_address, d10v_frame_saved_pc,
d10v_saved_pc_after_call, d10v_pop_frame, d10v_skip_prologue,
d10v_frame_chain, d10v_frame_init_saved_regs,
d10v_init_extra_frame_info, d10v_read_pc, d10v_write_pc,
d10v_read_sp, d10v_write_sp, d10v_write_fp, d10v_read_fp,
d10v_push_return_address, d10v_push_arguments,
d10v_extract_return_value): Make these functions static.
2001-06-29 06:10:41 +08:00
|
|
|
static CORE_ADDR
|
2000-07-30 09:48:28 +08:00
|
|
|
d10v_skip_prologue (CORE_ADDR pc)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
|
|
|
unsigned long op;
|
|
|
|
unsigned short op1, op2;
|
|
|
|
CORE_ADDR func_addr, func_end;
|
|
|
|
struct symtab_and_line sal;
|
|
|
|
|
2003-10-10 08:25:43 +08:00
|
|
|
/* If we have line debugging information, then the end of the prologue
|
2003-10-11 05:59:05 +08:00
|
|
|
should be the first assembly instruction of the first source line. */
|
1999-04-16 09:35:26 +08:00
|
|
|
if (find_pc_partial_function (pc, NULL, &func_addr, &func_end))
|
|
|
|
{
|
|
|
|
sal = find_pc_line (func_addr, 0);
|
1999-07-08 04:19:36 +08:00
|
|
|
if (sal.end && sal.end < func_end)
|
1999-04-16 09:35:26 +08:00
|
|
|
return sal.end;
|
|
|
|
}
|
1999-07-08 04:19:36 +08:00
|
|
|
|
|
|
|
if (target_read_memory (pc, (char *) &op, 4))
|
2003-10-10 08:25:43 +08:00
|
|
|
return pc; /* Can't access it -- assume no prologue. */
|
1999-04-16 09:35:26 +08:00
|
|
|
|
|
|
|
while (1)
|
|
|
|
{
|
1999-07-08 04:19:36 +08:00
|
|
|
op = (unsigned long) read_memory_integer (pc, 4);
|
1999-04-16 09:35:26 +08:00
|
|
|
if ((op & 0xC0000000) == 0xC0000000)
|
|
|
|
{
|
|
|
|
/* long instruction */
|
1999-07-08 04:19:36 +08:00
|
|
|
if (((op & 0x3FFF0000) != 0x01FF0000) && /* add3 sp,sp,n */
|
|
|
|
((op & 0x3F0F0000) != 0x340F0000) && /* st rn, @(offset,sp) */
|
|
|
|
((op & 0x3F1F0000) != 0x350F0000)) /* st2w rn, @(offset,sp) */
|
1999-04-16 09:35:26 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* short instructions */
|
|
|
|
if ((op & 0xC0000000) == 0x80000000)
|
|
|
|
{
|
|
|
|
op2 = (op & 0x3FFF8000) >> 15;
|
|
|
|
op1 = op & 0x7FFF;
|
1999-07-08 04:19:36 +08:00
|
|
|
}
|
|
|
|
else
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
|
|
|
op1 = (op & 0x3FFF8000) >> 15;
|
|
|
|
op2 = op & 0x7FFF;
|
|
|
|
}
|
1999-07-08 04:19:36 +08:00
|
|
|
if (check_prologue (op1))
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
1999-07-08 04:19:36 +08:00
|
|
|
if (!check_prologue (op2))
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
2003-10-10 08:25:43 +08:00
|
|
|
/* If the previous opcode was really part of the
|
|
|
|
prologue and not just a NOP, then we want to
|
|
|
|
break after both instructions. */
|
1999-04-16 09:35:26 +08:00
|
|
|
if (op1 != 0x5E00)
|
|
|
|
pc += 4;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
pc += 4;
|
|
|
|
}
|
|
|
|
return pc;
|
|
|
|
}
|
|
|
|
|
2003-01-30 23:11:20 +08:00
|
|
|
struct d10v_unwind_cache
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
2003-04-02 03:55:03 +08:00
|
|
|
/* The previous frame's inner most stack address. Used as this
|
|
|
|
frame ID's stack_addr. */
|
|
|
|
CORE_ADDR prev_sp;
|
|
|
|
/* The frame's base, optionally used by the high-level debug info. */
|
2003-03-06 06:54:22 +08:00
|
|
|
CORE_ADDR base;
|
2003-01-30 23:11:20 +08:00
|
|
|
int size;
|
2003-03-06 04:57:28 +08:00
|
|
|
/* How far the SP and r11 (FP) have been offset from the start of
|
|
|
|
the stack frame (as defined by the previous frame's stack
|
|
|
|
pointer). */
|
|
|
|
LONGEST sp_offset;
|
|
|
|
LONGEST r11_offset;
|
2003-01-30 23:11:20 +08:00
|
|
|
int uses_frame;
|
2003-06-09 00:09:46 +08:00
|
|
|
/* Table indicating the location of each and every register. */
|
2003-06-09 06:10:12 +08:00
|
|
|
struct trad_frame_saved_reg *saved_regs;
|
2003-01-30 23:11:20 +08:00
|
|
|
};
|
1999-04-16 09:35:26 +08:00
|
|
|
|
1999-07-08 04:19:36 +08:00
|
|
|
static int
|
2003-01-30 23:11:20 +08:00
|
|
|
prologue_find_regs (struct d10v_unwind_cache *info, unsigned short op,
|
|
|
|
CORE_ADDR addr)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
|
|
|
int n;
|
|
|
|
|
|
|
|
/* st rn, @-sp */
|
|
|
|
if ((op & 0x7E1F) == 0x6C1F)
|
|
|
|
{
|
|
|
|
n = (op & 0x1E0) >> 5;
|
2003-03-06 04:57:28 +08:00
|
|
|
info->sp_offset -= 2;
|
2003-06-09 00:09:46 +08:00
|
|
|
info->saved_regs[n].addr = info->sp_offset;
|
1999-04-16 09:35:26 +08:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* st2w rn, @-sp */
|
|
|
|
else if ((op & 0x7E3F) == 0x6E1F)
|
|
|
|
{
|
|
|
|
n = (op & 0x1E0) >> 5;
|
2003-03-06 04:57:28 +08:00
|
|
|
info->sp_offset -= 4;
|
2003-06-09 00:09:46 +08:00
|
|
|
info->saved_regs[n + 0].addr = info->sp_offset + 0;
|
|
|
|
info->saved_regs[n + 1].addr = info->sp_offset + 2;
|
1999-04-16 09:35:26 +08:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* subi sp, n */
|
|
|
|
if ((op & 0x7FE1) == 0x01E1)
|
|
|
|
{
|
|
|
|
n = (op & 0x1E) >> 1;
|
|
|
|
if (n == 0)
|
|
|
|
n = 16;
|
2003-03-06 04:57:28 +08:00
|
|
|
info->sp_offset -= n;
|
1999-04-16 09:35:26 +08:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* mv r11, sp */
|
|
|
|
if (op == 0x417E)
|
|
|
|
{
|
2003-01-30 23:11:20 +08:00
|
|
|
info->uses_frame = 1;
|
2003-03-06 04:57:28 +08:00
|
|
|
info->r11_offset = info->sp_offset;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* st rn, @r11 */
|
|
|
|
if ((op & 0x7E1F) == 0x6816)
|
|
|
|
{
|
|
|
|
n = (op & 0x1E0) >> 5;
|
2003-06-09 00:09:46 +08:00
|
|
|
info->saved_regs[n].addr = info->r11_offset;
|
1999-04-16 09:35:26 +08:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* nop */
|
|
|
|
if (op == 0x5E00)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
/* st rn, @sp */
|
|
|
|
if ((op & 0x7E1F) == 0x681E)
|
|
|
|
{
|
|
|
|
n = (op & 0x1E0) >> 5;
|
2003-06-09 00:09:46 +08:00
|
|
|
info->saved_regs[n].addr = info->sp_offset;
|
1999-04-16 09:35:26 +08:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* st2w rn, @sp */
|
|
|
|
if ((op & 0x7E3F) == 0x3A1E)
|
|
|
|
{
|
|
|
|
n = (op & 0x1E0) >> 5;
|
2003-06-09 00:09:46 +08:00
|
|
|
info->saved_regs[n + 0].addr = info->sp_offset + 0;
|
|
|
|
info->saved_regs[n + 1].addr = info->sp_offset + 2;
|
1999-04-16 09:35:26 +08:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
1999-06-08 03:19:32 +08:00
|
|
|
/* Put here the code to store, into fi->saved_regs, the addresses of
|
|
|
|
the saved registers of frame described by FRAME_INFO. This
|
|
|
|
includes special registers such as pc and fp saved in special ways
|
|
|
|
in the stack frame. sp is even more special: the address we return
|
2003-10-10 08:25:43 +08:00
|
|
|
for it IS the sp for the next frame. */
|
1999-06-08 03:19:32 +08:00
|
|
|
|
2003-06-08 Andrew Cagney <cagney@redhat.com>
* acinclude.m4 (gcc_AC_CHECK_DECL, (gcc_AC_CHECK_DECL): Stolen
from GCC's acinclude.m4.
* configure.in: Check for getopt's delcaration.
* aclocal.m4, config.in, configure: Re-generate.
* main.c (error_init): Delete declaration.
* defs.h (error_init): Declare.
* rs6000-tdep.c (rs6000_fetch_pointer_argument): Make static.
(rs6000_convert_from_func_ptr_addr): Make static.
(_initialize_rs6000_tdep): Add declaration.
* cli/cli-cmds.c (dont_repeat): Delete declaration.
(show_commands, set_verbose, show_history): Delete declaration.
* top.h (set_verbose): Add declaration.
(show_history, set_history, show_commands): Add declaration.
(do_restore_instream_cleanup): Add declaration.
* objc-lang.c (specialcmp): Make static.
(print_object_command): Make static.
(find_objc_msgsend): Make static.
(find_objc_msgcall_submethod_helper): Make static.
(find_objc_msgcall_submethod): Make static.
(_initialize_objc_language): Add declaration.
(find_implementation_from_class): Make static.
(find_implementation): Make static.
* objc-exp.y (yylex): Delete lookup_struct_typedef declaration.
* objc-lang.h (lookup_struct_typedef): Add declaration.
* cli/cli-interp.c (_initialize_cli_interp): Add declaration.
* cli/cli-script.c (clear_hook_in_cleanup): Make static.
(do_restore_user_call_depth): Make static.
(do_restore_instream_cleanup): Delete declaration.
(dont_repeat): Delete declaration.
* cli/cli-decode.c (add_abbrev_cmd): Delete function.
* cli/cli-dump.c (_initialize_cli_dump): Add declaration.
* reggroups.c (_initialize_reggroup): Add declaration.
* cp-support.c (_initialize_cp_support): Add declaration.
* cp-abi.c (_initialize_cp_abi): Add declaration.
* hpacc-abi.c (_initialize_hpacc_abi): Add declaration.
* gnu-v3-abi.c (gnuv3_baseclass_offset): Make static.
(_initialize_gnu_v3_abi): Add declaration.
* gnu-v2-abi.c (gnuv2_value_rtti_type): Make static.
(_initialize_gnu_v2_abi): Add declaration.
* frame-base.c (_initialize_frame_base): Add declaration.
* doublest.c (floatformat_from_length): Make static.
* frame-unwind.c (_initialize_frame_unwind): Add declaration.
* frame.c (create_sentinel_frame): Make static.
(_initialize_frame): Add declaration.
* top.c (do_catch_errors): Make static.
(gdb_rl_operate_and_get_next_completion): Make static.
* typeprint.c: Include "typeprint.h".
* sentinel-frame.c (sentinel_frame_prev_register): Make static.
(sentinel_frame_this_id): Make static.
* p-valprint.c (_initialize_pascal_valprint): Add declaration.
* ui-out.c (make_cleanup_ui_out_begin_end): Delete function.
* dwarf2-frame.c (dwarf2_frame_cache): Make static.
* p-exp.y (push_current_type, pop_current_type): ISO C declaration.
* dwarf2expr.h (dwarf_expr_context): ISO C declaration.
* maint.c (maintenance_print_architecture): Make static.
* signals/signals.c (_initialize_signals): Add declaration.
* std-regs.c (_initialize_frame_reg): Add declaration.
* jv-exp.y (push_variable): ISO C definition.
(push_qualified_expression_name): Ditto.
* memattr.c (_initialize_mem): Add declaration.
* remote.c (remote_check_watch_resources): Make static.
(remote_stopped_by_watchpoint): Make static.
(remote_stopped_data_address): Make static.
* d10v-tdep.c (nr_dmap_regs): Make static.
(a0_regnum): Make static.
(d10v_frame_unwind_cache): Make static.
(d10v_frame_p): Make static.
* osabi.c (show_osabi): Make static.
(_initialize_gdb_osabi): Add extern declaration.
* gdbtypes.c (make_qualified_type): Make static.
(safe_parse_type): Make static.
* macrocmd.c (_initialize_macrocmd): Add extern declaration.
* macrotab.c (macro_bcache_free): Make static.
* interps.c (interp_set_quiet): Make static.
(interpreter_exec_cmd): Make static.
* stack.h (select_frame_command): New file.
* stack.c: Include "stack.h".
(select_frame_command_wrapper): Delete function.
(select_frame_command): Make global.
* infcall.c: Include "infcall.h".
* linespec.c: Include "linespec.h".
* symfile.c (sections_overlap): Make static.
* cp-support.h (cp_initialize_namespace): ISO C declaration.
* charset.c (_initialize_charset): Add missing prototype.
* regcache.c (init_legacy_regcache_descr): Make static.
(do_regcache_xfree): Make static.
(regcache_xfer_part): Make static.
(_initialize_regcache): Add missing prototype.
* breakpoint.c (parse_breakpoint_sals): Make static.
(breakpoint_sals_to_pc): Make static.
* interps.h (clear_interpreter_hooks): ISO C declaration.
* Makefile.in (stack_h): Define.
(stack.o, typeprint.o, mi-main.o): Update dependencies.
(mi-cmd-stack.o, infcall.o, linespec.o): Update dependencies.
Index: mi/ChangeLog
2003-06-08 Andrew Cagney <cagney@redhat.com>
* mi-parse.c (_initialize_mi_parse): Delete function.
* mi-main.c: Include "mi-main.h".
* mi-interp.c (_initialize_mi_interp): Add declaration.
* mi-cmd-stack.c: Include "stack.h".
(select_frame_command_wrapper): Delete extern declaration.
(mi_cmd_stack_select_frame): Replace select_frame_command_wrapper
with select_frame_command.
2003-06-09 02:27:14 +08:00
|
|
|
static struct d10v_unwind_cache *
|
2003-03-17 22:23:50 +08:00
|
|
|
d10v_frame_unwind_cache (struct frame_info *next_frame,
|
|
|
|
void **this_prologue_cache)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
2003-06-04 02:53:37 +08:00
|
|
|
struct gdbarch *gdbarch = get_frame_arch (next_frame);
|
2003-03-06 06:54:22 +08:00
|
|
|
CORE_ADDR pc;
|
|
|
|
ULONGEST prev_sp;
|
|
|
|
ULONGEST this_base;
|
1999-04-16 09:35:26 +08:00
|
|
|
unsigned long op;
|
|
|
|
unsigned short op1, op2;
|
|
|
|
int i;
|
2003-01-30 23:11:20 +08:00
|
|
|
struct d10v_unwind_cache *info;
|
|
|
|
|
2003-03-17 22:23:50 +08:00
|
|
|
if ((*this_prologue_cache))
|
|
|
|
return (*this_prologue_cache);
|
2003-01-30 23:11:20 +08:00
|
|
|
|
|
|
|
info = FRAME_OBSTACK_ZALLOC (struct d10v_unwind_cache);
|
2003-03-17 22:23:50 +08:00
|
|
|
(*this_prologue_cache) = info;
|
2003-06-09 00:09:46 +08:00
|
|
|
info->saved_regs = trad_frame_alloc_saved_regs (next_frame);
|
2003-01-30 23:11:20 +08:00
|
|
|
|
|
|
|
info->size = 0;
|
2003-03-06 04:57:28 +08:00
|
|
|
info->sp_offset = 0;
|
1999-04-16 09:35:26 +08:00
|
|
|
|
2003-01-30 23:11:20 +08:00
|
|
|
info->uses_frame = 0;
|
2003-04-06 02:54:38 +08:00
|
|
|
for (pc = frame_func_unwind (next_frame);
|
2003-04-11 11:12:58 +08:00
|
|
|
pc > 0 && pc < frame_pc_unwind (next_frame);
|
2003-04-02 03:55:03 +08:00
|
|
|
pc += 4)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
2003-06-04 02:53:37 +08:00
|
|
|
op = get_frame_memory_unsigned (next_frame, pc, 4);
|
1999-04-16 09:35:26 +08:00
|
|
|
if ((op & 0xC0000000) == 0xC0000000)
|
|
|
|
{
|
|
|
|
/* long instruction */
|
|
|
|
if ((op & 0x3FFF0000) == 0x01FF0000)
|
|
|
|
{
|
|
|
|
/* add3 sp,sp,n */
|
|
|
|
short n = op & 0xFFFF;
|
2003-03-06 04:57:28 +08:00
|
|
|
info->sp_offset += n;
|
1999-04-16 09:35:26 +08:00
|
|
|
}
|
|
|
|
else if ((op & 0x3F0F0000) == 0x340F0000)
|
|
|
|
{
|
|
|
|
/* st rn, @(offset,sp) */
|
|
|
|
short offset = op & 0xFFFF;
|
|
|
|
short n = (op >> 20) & 0xF;
|
2003-06-09 00:09:46 +08:00
|
|
|
info->saved_regs[n].addr = info->sp_offset + offset;
|
1999-04-16 09:35:26 +08:00
|
|
|
}
|
|
|
|
else if ((op & 0x3F1F0000) == 0x350F0000)
|
|
|
|
{
|
|
|
|
/* st2w rn, @(offset,sp) */
|
|
|
|
short offset = op & 0xFFFF;
|
|
|
|
short n = (op >> 20) & 0xF;
|
2003-06-09 00:09:46 +08:00
|
|
|
info->saved_regs[n + 0].addr = info->sp_offset + offset + 0;
|
|
|
|
info->saved_regs[n + 1].addr = info->sp_offset + offset + 2;
|
1999-04-16 09:35:26 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* short instructions */
|
|
|
|
if ((op & 0xC0000000) == 0x80000000)
|
|
|
|
{
|
|
|
|
op2 = (op & 0x3FFF8000) >> 15;
|
|
|
|
op1 = op & 0x7FFF;
|
1999-07-08 04:19:36 +08:00
|
|
|
}
|
|
|
|
else
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
|
|
|
op1 = (op & 0x3FFF8000) >> 15;
|
|
|
|
op2 = op & 0x7FFF;
|
|
|
|
}
|
2003-01-30 23:11:20 +08:00
|
|
|
if (!prologue_find_regs (info, op1, pc)
|
|
|
|
|| !prologue_find_regs (info, op2, pc))
|
1999-04-16 09:35:26 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
1999-07-08 04:19:36 +08:00
|
|
|
|
2003-03-06 04:57:28 +08:00
|
|
|
info->size = -info->sp_offset;
|
1999-04-16 09:35:26 +08:00
|
|
|
|
2003-06-09 00:09:46 +08:00
|
|
|
/* Compute the previous frame's stack pointer (which is also the
|
|
|
|
frame's ID's stack address), and this frame's base pointer. */
|
2003-03-06 06:54:22 +08:00
|
|
|
if (info->uses_frame)
|
|
|
|
{
|
|
|
|
/* The SP was moved to the FP. This indicates that a new frame
|
|
|
|
was created. Get THIS frame's FP value by unwinding it from
|
|
|
|
the next frame. */
|
2003-03-31 10:36:59 +08:00
|
|
|
frame_unwind_unsigned_register (next_frame, D10V_FP_REGNUM, &this_base);
|
2003-03-06 06:54:22 +08:00
|
|
|
/* The FP points at the last saved register. Adjust the FP back
|
|
|
|
to before the first saved register giving the SP. */
|
|
|
|
prev_sp = this_base + info->size;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Assume that the FP is this frame's SP but with that pushed
|
|
|
|
stack space added back. */
|
2003-04-26 05:20:58 +08:00
|
|
|
frame_unwind_unsigned_register (next_frame, D10V_SP_REGNUM, &this_base);
|
2003-03-06 06:54:22 +08:00
|
|
|
prev_sp = this_base + info->size;
|
|
|
|
}
|
|
|
|
|
2003-06-09 00:09:46 +08:00
|
|
|
/* Convert that SP/BASE into real addresses. */
|
|
|
|
info->prev_sp = d10v_make_daddr (prev_sp);
|
2003-03-06 06:54:22 +08:00
|
|
|
info->base = d10v_make_daddr (this_base);
|
1999-04-16 09:35:26 +08:00
|
|
|
|
2003-03-06 06:54:22 +08:00
|
|
|
/* Adjust all the saved registers so that they contain addresses and
|
|
|
|
not offsets. */
|
1999-07-08 04:19:36 +08:00
|
|
|
for (i = 0; i < NUM_REGS - 1; i++)
|
2003-07-01 21:25:19 +08:00
|
|
|
if (trad_frame_addr_p (info->saved_regs, i))
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
2003-06-09 00:09:46 +08:00
|
|
|
info->saved_regs[i].addr = (info->prev_sp + info->saved_regs[i].addr);
|
1999-04-16 09:35:26 +08:00
|
|
|
}
|
|
|
|
|
2003-06-09 00:09:46 +08:00
|
|
|
/* The call instruction moves the caller's PC in the callee's LR.
|
|
|
|
Since this is an unwind, do the reverse. Copy the location of LR
|
|
|
|
into PC (the address / regnum) so that a request for PC will be
|
|
|
|
converted into a request for the LR. */
|
|
|
|
info->saved_regs[D10V_PC_REGNUM] = info->saved_regs[LR_REGNUM];
|
|
|
|
|
|
|
|
/* The previous frame's SP needed to be computed. Save the computed
|
|
|
|
value. */
|
2003-07-01 21:25:19 +08:00
|
|
|
trad_frame_set_value (info->saved_regs, D10V_SP_REGNUM,
|
|
|
|
d10v_make_daddr (prev_sp));
|
1999-04-16 09:35:26 +08:00
|
|
|
|
2003-01-30 23:11:20 +08:00
|
|
|
return info;
|
1999-04-16 09:35:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2003-02-03 02:53:22 +08:00
|
|
|
d10v_print_registers_info (struct gdbarch *gdbarch, struct ui_file *file,
|
|
|
|
struct frame_info *frame, int regnum, int all)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
2003-05-08 03:21:13 +08:00
|
|
|
struct gdbarch_tdep *tdep = gdbarch_tdep (gdbarch);
|
2003-02-03 02:53:22 +08:00
|
|
|
if (regnum >= 0)
|
1999-11-17 10:31:06 +08:00
|
|
|
{
|
2003-02-03 02:53:22 +08:00
|
|
|
default_print_registers_info (gdbarch, file, frame, regnum, all);
|
|
|
|
return;
|
1999-11-17 10:31:06 +08:00
|
|
|
}
|
2003-02-03 02:53:22 +08:00
|
|
|
|
|
|
|
{
|
|
|
|
ULONGEST pc, psw, rpt_s, rpt_e, rpt_c;
|
2003-09-29 06:32:20 +08:00
|
|
|
pc = get_frame_register_unsigned (frame, D10V_PC_REGNUM);
|
|
|
|
psw = get_frame_register_unsigned (frame, PSW_REGNUM);
|
|
|
|
rpt_s = get_frame_register_unsigned (frame, frame_map_name_to_regnum (frame, "rpt_s", -1));
|
|
|
|
rpt_e = get_frame_register_unsigned (frame, frame_map_name_to_regnum (frame, "rpt_e", -1));
|
|
|
|
rpt_c = get_frame_register_unsigned (frame, frame_map_name_to_regnum (frame, "rpt_c", -1));
|
2003-02-03 02:53:22 +08:00
|
|
|
fprintf_filtered (file, "PC=%04lx (0x%lx) PSW=%04lx RPT_S=%04lx RPT_E=%04lx RPT_C=%04lx\n",
|
|
|
|
(long) pc, (long) d10v_make_iaddr (pc), (long) psw,
|
|
|
|
(long) rpt_s, (long) rpt_e, (long) rpt_c);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
int group;
|
|
|
|
for (group = 0; group < 16; group += 8)
|
|
|
|
{
|
|
|
|
int r;
|
|
|
|
fprintf_filtered (file, "R%d-R%-2d", group, group + 7);
|
|
|
|
for (r = group; r < group + 8; r++)
|
|
|
|
{
|
|
|
|
ULONGEST tmp;
|
2003-09-29 06:32:20 +08:00
|
|
|
tmp = get_frame_register_unsigned (frame, r);
|
2003-02-03 02:53:22 +08:00
|
|
|
fprintf_filtered (file, " %04lx", (long) tmp);
|
|
|
|
}
|
|
|
|
fprintf_filtered (file, "\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Note: The IMAP/DMAP registers don't participate in function
|
|
|
|
calls. Don't bother trying to unwind them. */
|
|
|
|
|
2003-02-02 13:46:14 +08:00
|
|
|
{
|
2003-02-03 02:53:22 +08:00
|
|
|
int a;
|
|
|
|
for (a = 0; a < NR_IMAP_REGS; a++)
|
|
|
|
{
|
|
|
|
if (a > 0)
|
|
|
|
fprintf_filtered (file, " ");
|
2003-05-08 03:21:13 +08:00
|
|
|
fprintf_filtered (file, "IMAP%d %04lx", a,
|
|
|
|
tdep->imap_register (current_regcache, a));
|
2003-02-03 02:53:22 +08:00
|
|
|
}
|
2003-04-26 05:20:58 +08:00
|
|
|
if (nr_dmap_regs (gdbarch) == 1)
|
2003-02-03 02:53:22 +08:00
|
|
|
/* Registers DMAP0 and DMAP1 are constant. Just return dmap2. */
|
2003-05-08 03:21:13 +08:00
|
|
|
fprintf_filtered (file, " DMAP %04lx\n",
|
|
|
|
tdep->dmap_register (current_regcache, 2));
|
2003-02-03 02:53:22 +08:00
|
|
|
else
|
|
|
|
{
|
2003-04-26 05:20:58 +08:00
|
|
|
for (a = 0; a < nr_dmap_regs (gdbarch); a++)
|
2003-02-03 02:53:22 +08:00
|
|
|
{
|
2003-05-08 03:21:13 +08:00
|
|
|
fprintf_filtered (file, " DMAP%d %04lx", a,
|
|
|
|
tdep->dmap_register (current_regcache, a));
|
2003-02-03 02:53:22 +08:00
|
|
|
}
|
|
|
|
fprintf_filtered (file, "\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
2003-05-09 02:46:49 +08:00
|
|
|
char num[MAX_REGISTER_SIZE];
|
2003-02-03 02:53:22 +08:00
|
|
|
int a;
|
|
|
|
fprintf_filtered (file, "A0-A%d", NR_A_REGS - 1);
|
2003-04-26 05:20:58 +08:00
|
|
|
for (a = a0_regnum (gdbarch); a < a0_regnum (gdbarch) + NR_A_REGS; a++)
|
2003-02-02 13:46:14 +08:00
|
|
|
{
|
|
|
|
int i;
|
2003-02-03 02:53:22 +08:00
|
|
|
fprintf_filtered (file, " ");
|
2003-09-29 06:32:20 +08:00
|
|
|
get_frame_register (frame, a, num);
|
2003-06-04 02:53:37 +08:00
|
|
|
for (i = 0; i < register_size (gdbarch, a); i++)
|
2003-02-02 13:46:14 +08:00
|
|
|
{
|
2003-02-03 02:53:22 +08:00
|
|
|
fprintf_filtered (file, "%02x", (num[i] & 0xff));
|
2003-02-02 13:46:14 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2003-02-03 02:53:22 +08:00
|
|
|
fprintf_filtered (file, "\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
show_regs (char *args, int from_tty)
|
|
|
|
{
|
|
|
|
d10v_print_registers_info (current_gdbarch, gdb_stdout,
|
|
|
|
get_current_frame (), -1, 1);
|
1999-04-16 09:35:26 +08:00
|
|
|
}
|
|
|
|
|
* d10v-tdep.c (10v_frame_chain_valid, d10v_use_struct_convention,
d10v_breakpoint_from_pc, d10v_register_byte,
d10v_register_raw_size, d10v_register_virtual_size,
d10v_register_virtual_type, d10v_register_convertible,
d10v_register_convert_to_virtual, d10v_register_convert_to_raw,
d10v_make_daddr, d10v_make_iaddr, d10v_daddr_p, d10v_iaddr_p,
d10v_convert_iaddr_to_raw, d10v_convert_daddr_to_raw,
d10v_store_struct_return, d10v_store_return_value,
d10v_extract_struct_value_address, d10v_frame_saved_pc,
d10v_saved_pc_after_call, d10v_pop_frame, d10v_skip_prologue,
d10v_frame_chain, d10v_frame_init_saved_regs,
d10v_init_extra_frame_info, d10v_read_pc, d10v_write_pc,
d10v_read_sp, d10v_write_sp, d10v_write_fp, d10v_read_fp,
d10v_push_return_address, d10v_push_arguments,
d10v_extract_return_value): Make these functions static.
2001-06-29 06:10:41 +08:00
|
|
|
static CORE_ADDR
|
2001-05-04 12:15:33 +08:00
|
|
|
d10v_read_pc (ptid_t ptid)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
2001-05-04 12:15:33 +08:00
|
|
|
ptid_t save_ptid;
|
1999-04-16 09:35:26 +08:00
|
|
|
CORE_ADDR pc;
|
|
|
|
CORE_ADDR retval;
|
|
|
|
|
2001-05-04 12:15:33 +08:00
|
|
|
save_ptid = inferior_ptid;
|
|
|
|
inferior_ptid = ptid;
|
2003-05-03 07:56:12 +08:00
|
|
|
pc = (int) read_register (D10V_PC_REGNUM);
|
2001-05-04 12:15:33 +08:00
|
|
|
inferior_ptid = save_ptid;
|
* d10v-tdep.c (d10v_frame_chain, d10v_frame_init_saved_regs,
show_regs, d10v_read_pc, d10v_write_pc, d10v_read_sp,
d10v_write_sp, d10v_write_fp, d10v_read_fp,
d10v_push_return_address): Call the functions d10v_make_daddr,
d10v_make_iaddr, d10v_convert_iaddr_to_raw, and
d10v_convert_daddr_to_raw, not the global macros D10V_MAKE_DADDR,
D10V_MAKE_IADDR, D10V_CONVERT_IADDR_TO_RAW, and
D10V_CONVERT_DADDR_TO_RAW.
2001-07-06 05:27:08 +08:00
|
|
|
retval = d10v_make_iaddr (pc);
|
1999-04-16 09:35:26 +08:00
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
* d10v-tdep.c (10v_frame_chain_valid, d10v_use_struct_convention,
d10v_breakpoint_from_pc, d10v_register_byte,
d10v_register_raw_size, d10v_register_virtual_size,
d10v_register_virtual_type, d10v_register_convertible,
d10v_register_convert_to_virtual, d10v_register_convert_to_raw,
d10v_make_daddr, d10v_make_iaddr, d10v_daddr_p, d10v_iaddr_p,
d10v_convert_iaddr_to_raw, d10v_convert_daddr_to_raw,
d10v_store_struct_return, d10v_store_return_value,
d10v_extract_struct_value_address, d10v_frame_saved_pc,
d10v_saved_pc_after_call, d10v_pop_frame, d10v_skip_prologue,
d10v_frame_chain, d10v_frame_init_saved_regs,
d10v_init_extra_frame_info, d10v_read_pc, d10v_write_pc,
d10v_read_sp, d10v_write_sp, d10v_write_fp, d10v_read_fp,
d10v_push_return_address, d10v_push_arguments,
d10v_extract_return_value): Make these functions static.
2001-06-29 06:10:41 +08:00
|
|
|
static void
|
2001-05-04 12:15:33 +08:00
|
|
|
d10v_write_pc (CORE_ADDR val, ptid_t ptid)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
2001-05-04 12:15:33 +08:00
|
|
|
ptid_t save_ptid;
|
1999-04-16 09:35:26 +08:00
|
|
|
|
2001-05-04 12:15:33 +08:00
|
|
|
save_ptid = inferior_ptid;
|
|
|
|
inferior_ptid = ptid;
|
2003-05-03 07:56:12 +08:00
|
|
|
write_register (D10V_PC_REGNUM, d10v_convert_iaddr_to_raw (val));
|
2001-05-04 12:15:33 +08:00
|
|
|
inferior_ptid = save_ptid;
|
1999-04-16 09:35:26 +08:00
|
|
|
}
|
|
|
|
|
* d10v-tdep.c (10v_frame_chain_valid, d10v_use_struct_convention,
d10v_breakpoint_from_pc, d10v_register_byte,
d10v_register_raw_size, d10v_register_virtual_size,
d10v_register_virtual_type, d10v_register_convertible,
d10v_register_convert_to_virtual, d10v_register_convert_to_raw,
d10v_make_daddr, d10v_make_iaddr, d10v_daddr_p, d10v_iaddr_p,
d10v_convert_iaddr_to_raw, d10v_convert_daddr_to_raw,
d10v_store_struct_return, d10v_store_return_value,
d10v_extract_struct_value_address, d10v_frame_saved_pc,
d10v_saved_pc_after_call, d10v_pop_frame, d10v_skip_prologue,
d10v_frame_chain, d10v_frame_init_saved_regs,
d10v_init_extra_frame_info, d10v_read_pc, d10v_write_pc,
d10v_read_sp, d10v_write_sp, d10v_write_fp, d10v_read_fp,
d10v_push_return_address, d10v_push_arguments,
d10v_extract_return_value): Make these functions static.
2001-06-29 06:10:41 +08:00
|
|
|
static CORE_ADDR
|
2003-06-09 09:02:07 +08:00
|
|
|
d10v_unwind_sp (struct gdbarch *gdbarch, struct frame_info *next_frame)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
2003-06-09 09:02:07 +08:00
|
|
|
ULONGEST sp;
|
|
|
|
frame_unwind_unsigned_register (next_frame, D10V_SP_REGNUM, &sp);
|
|
|
|
return d10v_make_daddr (sp);
|
1999-04-16 09:35:26 +08:00
|
|
|
}
|
|
|
|
|
1999-04-27 02:34:20 +08:00
|
|
|
/* When arguments must be pushed onto the stack, they go on in reverse
|
2003-10-10 08:25:43 +08:00
|
|
|
order. The below implements a FILO (stack) to do this. */
|
1999-04-27 02:34:20 +08:00
|
|
|
|
|
|
|
struct stack_item
|
|
|
|
{
|
|
|
|
int len;
|
|
|
|
struct stack_item *prev;
|
|
|
|
void *data;
|
|
|
|
};
|
|
|
|
|
2000-05-28 09:12:42 +08:00
|
|
|
static struct stack_item *push_stack_item (struct stack_item *prev,
|
|
|
|
void *contents, int len);
|
1999-04-27 02:34:20 +08:00
|
|
|
static struct stack_item *
|
2000-07-30 09:48:28 +08:00
|
|
|
push_stack_item (struct stack_item *prev, void *contents, int len)
|
1999-04-27 02:34:20 +08:00
|
|
|
{
|
|
|
|
struct stack_item *si;
|
|
|
|
si = xmalloc (sizeof (struct stack_item));
|
|
|
|
si->data = xmalloc (len);
|
|
|
|
si->len = len;
|
|
|
|
si->prev = prev;
|
|
|
|
memcpy (si->data, contents, len);
|
|
|
|
return si;
|
|
|
|
}
|
|
|
|
|
2000-05-28 09:12:42 +08:00
|
|
|
static struct stack_item *pop_stack_item (struct stack_item *si);
|
1999-04-27 02:34:20 +08:00
|
|
|
static struct stack_item *
|
2000-07-30 09:48:28 +08:00
|
|
|
pop_stack_item (struct stack_item *si)
|
1999-04-27 02:34:20 +08:00
|
|
|
{
|
|
|
|
struct stack_item *dead = si;
|
|
|
|
si = si->prev;
|
2000-12-15 09:01:51 +08:00
|
|
|
xfree (dead->data);
|
|
|
|
xfree (dead);
|
1999-04-27 02:34:20 +08:00
|
|
|
return si;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-05-04 03:39:23 +08:00
|
|
|
static CORE_ADDR
|
|
|
|
d10v_push_dummy_code (struct gdbarch *gdbarch,
|
|
|
|
CORE_ADDR sp, CORE_ADDR funaddr, int using_gcc,
|
|
|
|
struct value **args, int nargs,
|
|
|
|
struct type *value_type,
|
|
|
|
CORE_ADDR *real_pc, CORE_ADDR *bp_addr)
|
|
|
|
{
|
|
|
|
/* Allocate space sufficient for a breakpoint. */
|
|
|
|
sp = (sp - 4) & ~3;
|
|
|
|
/* Store the address of that breakpoint taking care to first convert
|
|
|
|
it into a code (IADDR) address from a stack (DADDR) address.
|
|
|
|
This of course assumes that the two virtual addresses map onto
|
|
|
|
the same real address. */
|
|
|
|
(*bp_addr) = d10v_make_iaddr (d10v_convert_iaddr_to_raw (sp));
|
|
|
|
/* d10v always starts the call at the callee's entry point. */
|
|
|
|
(*real_pc) = funaddr;
|
|
|
|
return sp;
|
|
|
|
}
|
|
|
|
|
* d10v-tdep.c (10v_frame_chain_valid, d10v_use_struct_convention,
d10v_breakpoint_from_pc, d10v_register_byte,
d10v_register_raw_size, d10v_register_virtual_size,
d10v_register_virtual_type, d10v_register_convertible,
d10v_register_convert_to_virtual, d10v_register_convert_to_raw,
d10v_make_daddr, d10v_make_iaddr, d10v_daddr_p, d10v_iaddr_p,
d10v_convert_iaddr_to_raw, d10v_convert_daddr_to_raw,
d10v_store_struct_return, d10v_store_return_value,
d10v_extract_struct_value_address, d10v_frame_saved_pc,
d10v_saved_pc_after_call, d10v_pop_frame, d10v_skip_prologue,
d10v_frame_chain, d10v_frame_init_saved_regs,
d10v_init_extra_frame_info, d10v_read_pc, d10v_write_pc,
d10v_read_sp, d10v_write_sp, d10v_write_fp, d10v_read_fp,
d10v_push_return_address, d10v_push_arguments,
d10v_extract_return_value): Make these functions static.
2001-06-29 06:10:41 +08:00
|
|
|
static CORE_ADDR
|
2003-05-31 22:20:31 +08:00
|
|
|
d10v_push_dummy_call (struct gdbarch *gdbarch, CORE_ADDR func_addr,
|
|
|
|
struct regcache *regcache, CORE_ADDR bp_addr,
|
2003-10-10 08:25:43 +08:00
|
|
|
int nargs, struct value **args, CORE_ADDR sp,
|
|
|
|
int struct_return, CORE_ADDR struct_addr)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
int regnum = ARG1_REGNUM;
|
1999-04-27 02:34:20 +08:00
|
|
|
struct stack_item *si = NULL;
|
2002-06-07 02:57:08 +08:00
|
|
|
long val;
|
|
|
|
|
2003-03-27 23:09:48 +08:00
|
|
|
/* Set the return address. For the d10v, the return breakpoint is
|
2003-05-31 22:20:31 +08:00
|
|
|
always at BP_ADDR. */
|
2003-03-27 23:09:48 +08:00
|
|
|
regcache_cooked_write_unsigned (regcache, LR_REGNUM,
|
2003-05-31 22:20:31 +08:00
|
|
|
d10v_convert_iaddr_to_raw (bp_addr));
|
2003-03-27 23:09:48 +08:00
|
|
|
|
2003-03-26 04:38:47 +08:00
|
|
|
/* If STRUCT_RETURN is true, then the struct return address (in
|
|
|
|
STRUCT_ADDR) will consume the first argument-passing register.
|
|
|
|
Both adjust the register count and store that value. */
|
2002-06-07 02:57:08 +08:00
|
|
|
if (struct_return)
|
2003-03-26 04:38:47 +08:00
|
|
|
{
|
2003-03-27 23:09:48 +08:00
|
|
|
regcache_cooked_write_unsigned (regcache, regnum, struct_addr);
|
2003-03-26 04:38:47 +08:00
|
|
|
regnum++;
|
|
|
|
}
|
1999-07-08 04:19:36 +08:00
|
|
|
|
1999-04-16 09:35:26 +08:00
|
|
|
/* Fill in registers and arg lists */
|
|
|
|
for (i = 0; i < nargs; i++)
|
|
|
|
{
|
2001-07-16 04:10:02 +08:00
|
|
|
struct value *arg = args[i];
|
1999-04-16 09:35:26 +08:00
|
|
|
struct type *type = check_typedef (VALUE_TYPE (arg));
|
|
|
|
char *contents = VALUE_CONTENTS (arg);
|
|
|
|
int len = TYPE_LENGTH (type);
|
2002-06-07 02:57:08 +08:00
|
|
|
int aligned_regnum = (regnum + 1) & ~1;
|
|
|
|
|
2002-05-24 07:53:56 +08:00
|
|
|
/* printf ("push: type=%d len=%d\n", TYPE_CODE (type), len); */
|
2002-06-07 02:57:08 +08:00
|
|
|
if (len <= 2 && regnum <= ARGN_REGNUM)
|
|
|
|
/* fits in a single register, do not align */
|
|
|
|
{
|
|
|
|
val = extract_unsigned_integer (contents, len);
|
2003-03-27 23:09:48 +08:00
|
|
|
regcache_cooked_write_unsigned (regcache, regnum++, val);
|
2002-06-07 02:57:08 +08:00
|
|
|
}
|
|
|
|
else if (len <= (ARGN_REGNUM - aligned_regnum + 1) * 2)
|
|
|
|
/* value fits in remaining registers, store keeping left
|
|
|
|
aligned */
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
2002-06-07 02:57:08 +08:00
|
|
|
int b;
|
|
|
|
regnum = aligned_regnum;
|
|
|
|
for (b = 0; b < (len & ~1); b += 2)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
2002-06-07 02:57:08 +08:00
|
|
|
val = extract_unsigned_integer (&contents[b], 2);
|
2003-03-27 23:09:48 +08:00
|
|
|
regcache_cooked_write_unsigned (regcache, regnum++, val);
|
1999-04-16 09:35:26 +08:00
|
|
|
}
|
2002-06-07 02:57:08 +08:00
|
|
|
if (b < len)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
2002-06-07 02:57:08 +08:00
|
|
|
val = extract_unsigned_integer (&contents[b], 1);
|
2003-03-27 23:09:48 +08:00
|
|
|
regcache_cooked_write_unsigned (regcache, regnum++, (val << 8));
|
1999-04-16 09:35:26 +08:00
|
|
|
}
|
|
|
|
}
|
2002-06-07 02:57:08 +08:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* arg will go onto stack */
|
|
|
|
regnum = ARGN_REGNUM + 1;
|
|
|
|
si = push_stack_item (si, contents, len);
|
|
|
|
}
|
1999-04-16 09:35:26 +08:00
|
|
|
}
|
1999-04-27 02:34:20 +08:00
|
|
|
|
|
|
|
while (si)
|
|
|
|
{
|
|
|
|
sp = (sp - si->len) & ~1;
|
|
|
|
write_memory (sp, si->data, si->len);
|
|
|
|
si = pop_stack_item (si);
|
|
|
|
}
|
1999-07-08 04:19:36 +08:00
|
|
|
|
2003-03-27 23:09:48 +08:00
|
|
|
/* Finally, update the SP register. */
|
2003-04-26 05:20:58 +08:00
|
|
|
regcache_cooked_write_unsigned (regcache, D10V_SP_REGNUM,
|
2003-03-27 23:09:48 +08:00
|
|
|
d10v_convert_daddr_to_raw (sp));
|
|
|
|
|
1999-04-16 09:35:26 +08:00
|
|
|
return sp;
|
|
|
|
}
|
|
|
|
|
1999-09-22 11:28:34 +08:00
|
|
|
/* Translate a GDB virtual ADDR/LEN into a format the remote target
|
|
|
|
understands. Returns number of bytes that can be transfered
|
1999-11-17 10:31:06 +08:00
|
|
|
starting at TARG_ADDR. Return ZERO if no bytes can be transfered
|
|
|
|
(segmentation fault). Since the simulator knows all about how the
|
2003-10-10 08:25:43 +08:00
|
|
|
VM system works, we just call that to do the translation. */
|
1999-09-22 11:28:34 +08:00
|
|
|
|
1999-11-17 10:31:06 +08:00
|
|
|
static void
|
2003-05-08 03:21:13 +08:00
|
|
|
remote_d10v_translate_xfer_address (struct gdbarch *gdbarch,
|
|
|
|
struct regcache *regcache,
|
|
|
|
CORE_ADDR memaddr, int nr_bytes,
|
1999-09-22 11:28:34 +08:00
|
|
|
CORE_ADDR *targ_addr, int *targ_len)
|
|
|
|
{
|
2003-05-08 03:21:13 +08:00
|
|
|
struct gdbarch_tdep *tdep = gdbarch_tdep (gdbarch);
|
1999-11-17 10:31:06 +08:00
|
|
|
long out_addr;
|
|
|
|
long out_len;
|
2003-05-08 03:21:13 +08:00
|
|
|
out_len = sim_d10v_translate_addr (memaddr, nr_bytes, &out_addr, regcache,
|
|
|
|
tdep->dmap_register, tdep->imap_register);
|
1999-11-17 10:31:06 +08:00
|
|
|
*targ_addr = out_addr;
|
|
|
|
*targ_len = out_len;
|
1999-09-22 11:28:34 +08:00
|
|
|
}
|
|
|
|
|
1999-11-17 10:31:06 +08:00
|
|
|
|
1999-04-16 09:35:26 +08:00
|
|
|
/* The following code implements access to, and display of, the D10V's
|
|
|
|
instruction trace buffer. The buffer consists of 64K or more
|
|
|
|
4-byte words of data, of which each words includes an 8-bit count,
|
|
|
|
an 8-bit segment number, and a 16-bit instruction address.
|
|
|
|
|
|
|
|
In theory, the trace buffer is continuously capturing instruction
|
|
|
|
data that the CPU presents on its "debug bus", but in practice, the
|
|
|
|
ROMified GDB stub only enables tracing when it continues or steps
|
|
|
|
the program, and stops tracing when the program stops; so it
|
|
|
|
actually works for GDB to read the buffer counter out of memory and
|
|
|
|
then read each trace word. The counter records where the tracing
|
|
|
|
stops, but there is no record of where it started, so we remember
|
|
|
|
the PC when we resumed and then search backwards in the trace
|
|
|
|
buffer for a word that includes that address. This is not perfect,
|
|
|
|
because you will miss trace data if the resumption PC is the target
|
|
|
|
of a branch. (The value of the buffer counter is semi-random, any
|
|
|
|
trace data from a previous program stop is gone.) */
|
|
|
|
|
|
|
|
/* The address of the last word recorded in the trace buffer. */
|
|
|
|
|
|
|
|
#define DBBC_ADDR (0xd80000)
|
|
|
|
|
|
|
|
/* The base of the trace buffer, at least for the "Board_0". */
|
|
|
|
|
|
|
|
#define TRACE_BUFFER_BASE (0xf40000)
|
|
|
|
|
2000-05-28 09:12:42 +08:00
|
|
|
static void trace_command (char *, int);
|
1999-04-16 09:35:26 +08:00
|
|
|
|
2000-05-28 09:12:42 +08:00
|
|
|
static void untrace_command (char *, int);
|
1999-04-16 09:35:26 +08:00
|
|
|
|
2000-05-28 09:12:42 +08:00
|
|
|
static void trace_info (char *, int);
|
1999-04-16 09:35:26 +08:00
|
|
|
|
2000-05-28 09:12:42 +08:00
|
|
|
static void tdisassemble_command (char *, int);
|
1999-04-16 09:35:26 +08:00
|
|
|
|
2000-05-28 09:12:42 +08:00
|
|
|
static void display_trace (int, int);
|
1999-04-16 09:35:26 +08:00
|
|
|
|
|
|
|
/* True when instruction traces are being collected. */
|
|
|
|
|
|
|
|
static int tracing;
|
|
|
|
|
|
|
|
/* Remembered PC. */
|
|
|
|
|
|
|
|
static CORE_ADDR last_pc;
|
|
|
|
|
|
|
|
/* True when trace output should be displayed whenever program stops. */
|
|
|
|
|
|
|
|
static int trace_display;
|
|
|
|
|
|
|
|
/* True when trace listing should include source lines. */
|
|
|
|
|
|
|
|
static int default_trace_show_source = 1;
|
|
|
|
|
1999-07-08 04:19:36 +08:00
|
|
|
struct trace_buffer
|
|
|
|
{
|
|
|
|
int size;
|
|
|
|
short *counts;
|
|
|
|
CORE_ADDR *addrs;
|
|
|
|
}
|
|
|
|
trace_data;
|
1999-04-16 09:35:26 +08:00
|
|
|
|
|
|
|
static void
|
2000-07-30 09:48:28 +08:00
|
|
|
trace_command (char *args, int from_tty)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
|
|
|
/* Clear the host-side trace buffer, allocating space if needed. */
|
|
|
|
trace_data.size = 0;
|
|
|
|
if (trace_data.counts == NULL)
|
2003-04-26 05:20:58 +08:00
|
|
|
trace_data.counts = XCALLOC (65536, short);
|
1999-04-16 09:35:26 +08:00
|
|
|
if (trace_data.addrs == NULL)
|
2003-04-26 05:20:58 +08:00
|
|
|
trace_data.addrs = XCALLOC (65536, CORE_ADDR);
|
1999-04-16 09:35:26 +08:00
|
|
|
|
|
|
|
tracing = 1;
|
|
|
|
|
|
|
|
printf_filtered ("Tracing is now on.\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2000-07-30 09:48:28 +08:00
|
|
|
untrace_command (char *args, int from_tty)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
|
|
|
tracing = 0;
|
|
|
|
|
|
|
|
printf_filtered ("Tracing is now off.\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2000-07-30 09:48:28 +08:00
|
|
|
trace_info (char *args, int from_tty)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (trace_data.size)
|
|
|
|
{
|
|
|
|
printf_filtered ("%d entries in trace buffer:\n", trace_data.size);
|
|
|
|
|
|
|
|
for (i = 0; i < trace_data.size; ++i)
|
|
|
|
{
|
1999-09-09 08:02:17 +08:00
|
|
|
printf_filtered ("%d: %d instruction%s at 0x%s\n",
|
|
|
|
i,
|
|
|
|
trace_data.counts[i],
|
1999-04-16 09:35:26 +08:00
|
|
|
(trace_data.counts[i] == 1 ? "" : "s"),
|
1999-09-09 08:02:17 +08:00
|
|
|
paddr_nz (trace_data.addrs[i]));
|
1999-04-16 09:35:26 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
printf_filtered ("No entries in trace buffer.\n");
|
|
|
|
|
|
|
|
printf_filtered ("Tracing is currently %s.\n", (tracing ? "on" : "off"));
|
|
|
|
}
|
|
|
|
|
1999-05-26 02:09:09 +08:00
|
|
|
static void
|
2000-07-30 09:48:28 +08:00
|
|
|
d10v_eva_prepare_to_trace (void)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
|
|
|
if (!tracing)
|
|
|
|
return;
|
|
|
|
|
2003-05-03 07:56:12 +08:00
|
|
|
last_pc = read_register (D10V_PC_REGNUM);
|
1999-04-16 09:35:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Collect trace data from the target board and format it into a form
|
|
|
|
more useful for display. */
|
|
|
|
|
1999-05-26 02:09:09 +08:00
|
|
|
static void
|
2000-07-30 09:48:28 +08:00
|
|
|
d10v_eva_get_trace_data (void)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
|
|
|
int count, i, j, oldsize;
|
|
|
|
int trace_addr, trace_seg, trace_cnt, next_cnt;
|
|
|
|
unsigned int last_trace, trace_word, next_word;
|
|
|
|
unsigned int *tmpspace;
|
|
|
|
|
|
|
|
if (!tracing)
|
|
|
|
return;
|
|
|
|
|
1999-07-08 04:19:36 +08:00
|
|
|
tmpspace = xmalloc (65536 * sizeof (unsigned int));
|
1999-04-16 09:35:26 +08:00
|
|
|
|
|
|
|
last_trace = read_memory_unsigned_integer (DBBC_ADDR, 2) << 2;
|
|
|
|
|
|
|
|
/* Collect buffer contents from the target, stopping when we reach
|
|
|
|
the word recorded when execution resumed. */
|
|
|
|
|
|
|
|
count = 0;
|
|
|
|
while (last_trace > 0)
|
|
|
|
{
|
|
|
|
QUIT;
|
|
|
|
trace_word =
|
|
|
|
read_memory_unsigned_integer (TRACE_BUFFER_BASE + last_trace, 4);
|
|
|
|
trace_addr = trace_word & 0xffff;
|
|
|
|
last_trace -= 4;
|
|
|
|
/* Ignore an apparently nonsensical entry. */
|
|
|
|
if (trace_addr == 0xffd5)
|
|
|
|
continue;
|
|
|
|
tmpspace[count++] = trace_word;
|
|
|
|
if (trace_addr == last_pc)
|
|
|
|
break;
|
|
|
|
if (count > 65535)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Move the data to the host-side trace buffer, adjusting counts to
|
|
|
|
include the last instruction executed and transforming the address
|
|
|
|
into something that GDB likes. */
|
|
|
|
|
|
|
|
for (i = 0; i < count; ++i)
|
|
|
|
{
|
|
|
|
trace_word = tmpspace[i];
|
|
|
|
next_word = ((i == 0) ? 0 : tmpspace[i - 1]);
|
|
|
|
trace_addr = trace_word & 0xffff;
|
|
|
|
next_cnt = (next_word >> 24) & 0xff;
|
|
|
|
j = trace_data.size + count - i - 1;
|
|
|
|
trace_data.addrs[j] = (trace_addr << 2) + 0x1000000;
|
|
|
|
trace_data.counts[j] = next_cnt + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
oldsize = trace_data.size;
|
|
|
|
trace_data.size += count;
|
|
|
|
|
2000-12-15 09:01:51 +08:00
|
|
|
xfree (tmpspace);
|
1999-04-16 09:35:26 +08:00
|
|
|
|
|
|
|
if (trace_display)
|
|
|
|
display_trace (oldsize, trace_data.size);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2000-07-30 09:48:28 +08:00
|
|
|
tdisassemble_command (char *arg, int from_tty)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
|
|
|
int i, count;
|
|
|
|
CORE_ADDR low, high;
|
|
|
|
|
|
|
|
if (!arg)
|
|
|
|
{
|
|
|
|
low = 0;
|
|
|
|
high = trace_data.size;
|
|
|
|
}
|
|
|
|
else
|
2003-03-29 09:40:01 +08:00
|
|
|
{
|
|
|
|
char *space_index = strchr (arg, ' ');
|
|
|
|
if (space_index == NULL)
|
|
|
|
{
|
|
|
|
low = parse_and_eval_address (arg);
|
|
|
|
high = low + 5;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Two arguments. */
|
|
|
|
*space_index = '\0';
|
|
|
|
low = parse_and_eval_address (arg);
|
|
|
|
high = parse_and_eval_address (space_index + 1);
|
|
|
|
if (high < low)
|
|
|
|
high = low;
|
|
|
|
}
|
1999-04-16 09:35:26 +08:00
|
|
|
}
|
|
|
|
|
2003-10-10 08:25:43 +08:00
|
|
|
printf_filtered ("Dump of trace from %s to %s:\n",
|
|
|
|
paddr_u (low), paddr_u (high));
|
1999-04-16 09:35:26 +08:00
|
|
|
|
|
|
|
display_trace (low, high);
|
|
|
|
|
|
|
|
printf_filtered ("End of trace dump.\n");
|
|
|
|
gdb_flush (gdb_stdout);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2000-07-30 09:48:28 +08:00
|
|
|
display_trace (int low, int high)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
|
|
|
int i, count, trace_show_source, first, suppress;
|
|
|
|
CORE_ADDR next_address;
|
|
|
|
|
|
|
|
trace_show_source = default_trace_show_source;
|
1999-07-08 04:19:36 +08:00
|
|
|
if (!have_full_symbols () && !have_partial_symbols ())
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
|
|
|
trace_show_source = 0;
|
|
|
|
printf_filtered ("No symbol table is loaded. Use the \"file\" command.\n");
|
|
|
|
printf_filtered ("Trace will not display any source.\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
first = 1;
|
|
|
|
suppress = 0;
|
|
|
|
for (i = low; i < high; ++i)
|
|
|
|
{
|
|
|
|
next_address = trace_data.addrs[i];
|
1999-07-08 04:19:36 +08:00
|
|
|
count = trace_data.counts[i];
|
1999-04-16 09:35:26 +08:00
|
|
|
while (count-- > 0)
|
|
|
|
{
|
|
|
|
QUIT;
|
|
|
|
if (trace_show_source)
|
|
|
|
{
|
|
|
|
struct symtab_and_line sal, sal_prev;
|
|
|
|
|
|
|
|
sal_prev = find_pc_line (next_address - 4, 0);
|
|
|
|
sal = find_pc_line (next_address, 0);
|
|
|
|
|
|
|
|
if (sal.symtab)
|
|
|
|
{
|
|
|
|
if (first || sal.line != sal_prev.line)
|
|
|
|
print_source_lines (sal.symtab, sal.line, sal.line + 1, 0);
|
|
|
|
suppress = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!suppress)
|
|
|
|
/* FIXME-32x64--assumes sal.pc fits in long. */
|
|
|
|
printf_filtered ("No source file for address %s.\n",
|
1999-07-08 04:19:36 +08:00
|
|
|
local_hex_string ((unsigned long) sal.pc));
|
1999-04-16 09:35:26 +08:00
|
|
|
suppress = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
first = 0;
|
|
|
|
print_address (next_address, gdb_stdout);
|
|
|
|
printf_filtered (":");
|
|
|
|
printf_filtered ("\t");
|
|
|
|
wrap_here (" ");
|
2003-05-04 03:13:04 +08:00
|
|
|
next_address += gdb_print_insn (next_address, gdb_stdout);
|
1999-04-16 09:35:26 +08:00
|
|
|
printf_filtered ("\n");
|
|
|
|
gdb_flush (gdb_stdout);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-01-30 23:11:20 +08:00
|
|
|
static CORE_ADDR
|
2003-03-10 23:28:41 +08:00
|
|
|
d10v_unwind_pc (struct gdbarch *gdbarch, struct frame_info *next_frame)
|
2003-01-30 23:11:20 +08:00
|
|
|
{
|
2003-03-10 23:28:41 +08:00
|
|
|
ULONGEST pc;
|
2003-05-03 07:56:12 +08:00
|
|
|
frame_unwind_unsigned_register (next_frame, D10V_PC_REGNUM, &pc);
|
2003-03-10 23:28:41 +08:00
|
|
|
return d10v_make_iaddr (pc);
|
2003-01-30 23:11:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Given a GDB frame, determine the address of the calling function's
|
|
|
|
frame. This will be used to create a new GDB frame struct. */
|
|
|
|
|
|
|
|
static void
|
2003-03-17 22:23:50 +08:00
|
|
|
d10v_frame_this_id (struct frame_info *next_frame,
|
|
|
|
void **this_prologue_cache,
|
|
|
|
struct frame_id *this_id)
|
2003-01-30 23:11:20 +08:00
|
|
|
{
|
2003-03-17 22:23:50 +08:00
|
|
|
struct d10v_unwind_cache *info
|
|
|
|
= d10v_frame_unwind_cache (next_frame, this_prologue_cache);
|
|
|
|
CORE_ADDR base;
|
2003-04-11 11:12:58 +08:00
|
|
|
CORE_ADDR func;
|
|
|
|
struct frame_id id;
|
2003-01-30 23:11:20 +08:00
|
|
|
|
2003-04-11 11:12:58 +08:00
|
|
|
/* The FUNC is easy. */
|
|
|
|
func = frame_func_unwind (next_frame);
|
2003-01-30 23:11:20 +08:00
|
|
|
|
2003-03-17 22:23:50 +08:00
|
|
|
/* Hopefully the prologue analysis either correctly determined the
|
|
|
|
frame's base (which is the SP from the previous frame), or set
|
|
|
|
that base to "NULL". */
|
2003-04-02 03:55:03 +08:00
|
|
|
base = info->prev_sp;
|
2003-03-17 22:23:50 +08:00
|
|
|
if (base == STACK_START || base == 0)
|
|
|
|
return;
|
2003-01-30 23:11:20 +08:00
|
|
|
|
2003-04-11 11:12:58 +08:00
|
|
|
id = frame_id_build (base, func);
|
|
|
|
|
|
|
|
(*this_id) = id;
|
2003-01-30 23:11:20 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2003-03-17 22:23:50 +08:00
|
|
|
d10v_frame_prev_register (struct frame_info *next_frame,
|
|
|
|
void **this_prologue_cache,
|
|
|
|
int regnum, int *optimizedp,
|
|
|
|
enum lval_type *lvalp, CORE_ADDR *addrp,
|
|
|
|
int *realnump, void *bufferp)
|
2003-01-30 23:11:20 +08:00
|
|
|
{
|
2003-03-17 22:23:50 +08:00
|
|
|
struct d10v_unwind_cache *info
|
|
|
|
= d10v_frame_unwind_cache (next_frame, this_prologue_cache);
|
2003-06-09 00:09:46 +08:00
|
|
|
trad_frame_prev_register (next_frame, info->saved_regs, regnum,
|
|
|
|
optimizedp, lvalp, addrp, realnump, bufferp);
|
2003-01-30 23:11:20 +08:00
|
|
|
}
|
|
|
|
|
2003-04-02 03:55:03 +08:00
|
|
|
static const struct frame_unwind d10v_frame_unwind = {
|
2003-04-05 11:56:00 +08:00
|
|
|
NORMAL_FRAME,
|
2003-03-17 22:23:50 +08:00
|
|
|
d10v_frame_this_id,
|
|
|
|
d10v_frame_prev_register
|
2003-01-30 23:11:20 +08:00
|
|
|
};
|
|
|
|
|
2003-06-08 Andrew Cagney <cagney@redhat.com>
* acinclude.m4 (gcc_AC_CHECK_DECL, (gcc_AC_CHECK_DECL): Stolen
from GCC's acinclude.m4.
* configure.in: Check for getopt's delcaration.
* aclocal.m4, config.in, configure: Re-generate.
* main.c (error_init): Delete declaration.
* defs.h (error_init): Declare.
* rs6000-tdep.c (rs6000_fetch_pointer_argument): Make static.
(rs6000_convert_from_func_ptr_addr): Make static.
(_initialize_rs6000_tdep): Add declaration.
* cli/cli-cmds.c (dont_repeat): Delete declaration.
(show_commands, set_verbose, show_history): Delete declaration.
* top.h (set_verbose): Add declaration.
(show_history, set_history, show_commands): Add declaration.
(do_restore_instream_cleanup): Add declaration.
* objc-lang.c (specialcmp): Make static.
(print_object_command): Make static.
(find_objc_msgsend): Make static.
(find_objc_msgcall_submethod_helper): Make static.
(find_objc_msgcall_submethod): Make static.
(_initialize_objc_language): Add declaration.
(find_implementation_from_class): Make static.
(find_implementation): Make static.
* objc-exp.y (yylex): Delete lookup_struct_typedef declaration.
* objc-lang.h (lookup_struct_typedef): Add declaration.
* cli/cli-interp.c (_initialize_cli_interp): Add declaration.
* cli/cli-script.c (clear_hook_in_cleanup): Make static.
(do_restore_user_call_depth): Make static.
(do_restore_instream_cleanup): Delete declaration.
(dont_repeat): Delete declaration.
* cli/cli-decode.c (add_abbrev_cmd): Delete function.
* cli/cli-dump.c (_initialize_cli_dump): Add declaration.
* reggroups.c (_initialize_reggroup): Add declaration.
* cp-support.c (_initialize_cp_support): Add declaration.
* cp-abi.c (_initialize_cp_abi): Add declaration.
* hpacc-abi.c (_initialize_hpacc_abi): Add declaration.
* gnu-v3-abi.c (gnuv3_baseclass_offset): Make static.
(_initialize_gnu_v3_abi): Add declaration.
* gnu-v2-abi.c (gnuv2_value_rtti_type): Make static.
(_initialize_gnu_v2_abi): Add declaration.
* frame-base.c (_initialize_frame_base): Add declaration.
* doublest.c (floatformat_from_length): Make static.
* frame-unwind.c (_initialize_frame_unwind): Add declaration.
* frame.c (create_sentinel_frame): Make static.
(_initialize_frame): Add declaration.
* top.c (do_catch_errors): Make static.
(gdb_rl_operate_and_get_next_completion): Make static.
* typeprint.c: Include "typeprint.h".
* sentinel-frame.c (sentinel_frame_prev_register): Make static.
(sentinel_frame_this_id): Make static.
* p-valprint.c (_initialize_pascal_valprint): Add declaration.
* ui-out.c (make_cleanup_ui_out_begin_end): Delete function.
* dwarf2-frame.c (dwarf2_frame_cache): Make static.
* p-exp.y (push_current_type, pop_current_type): ISO C declaration.
* dwarf2expr.h (dwarf_expr_context): ISO C declaration.
* maint.c (maintenance_print_architecture): Make static.
* signals/signals.c (_initialize_signals): Add declaration.
* std-regs.c (_initialize_frame_reg): Add declaration.
* jv-exp.y (push_variable): ISO C definition.
(push_qualified_expression_name): Ditto.
* memattr.c (_initialize_mem): Add declaration.
* remote.c (remote_check_watch_resources): Make static.
(remote_stopped_by_watchpoint): Make static.
(remote_stopped_data_address): Make static.
* d10v-tdep.c (nr_dmap_regs): Make static.
(a0_regnum): Make static.
(d10v_frame_unwind_cache): Make static.
(d10v_frame_p): Make static.
* osabi.c (show_osabi): Make static.
(_initialize_gdb_osabi): Add extern declaration.
* gdbtypes.c (make_qualified_type): Make static.
(safe_parse_type): Make static.
* macrocmd.c (_initialize_macrocmd): Add extern declaration.
* macrotab.c (macro_bcache_free): Make static.
* interps.c (interp_set_quiet): Make static.
(interpreter_exec_cmd): Make static.
* stack.h (select_frame_command): New file.
* stack.c: Include "stack.h".
(select_frame_command_wrapper): Delete function.
(select_frame_command): Make global.
* infcall.c: Include "infcall.h".
* linespec.c: Include "linespec.h".
* symfile.c (sections_overlap): Make static.
* cp-support.h (cp_initialize_namespace): ISO C declaration.
* charset.c (_initialize_charset): Add missing prototype.
* regcache.c (init_legacy_regcache_descr): Make static.
(do_regcache_xfree): Make static.
(regcache_xfer_part): Make static.
(_initialize_regcache): Add missing prototype.
* breakpoint.c (parse_breakpoint_sals): Make static.
(breakpoint_sals_to_pc): Make static.
* interps.h (clear_interpreter_hooks): ISO C declaration.
* Makefile.in (stack_h): Define.
(stack.o, typeprint.o, mi-main.o): Update dependencies.
(mi-cmd-stack.o, infcall.o, linespec.o): Update dependencies.
Index: mi/ChangeLog
2003-06-08 Andrew Cagney <cagney@redhat.com>
* mi-parse.c (_initialize_mi_parse): Delete function.
* mi-main.c: Include "mi-main.h".
* mi-interp.c (_initialize_mi_interp): Add declaration.
* mi-cmd-stack.c: Include "stack.h".
(select_frame_command_wrapper): Delete extern declaration.
(mi_cmd_stack_select_frame): Replace select_frame_command_wrapper
with select_frame_command.
2003-06-09 02:27:14 +08:00
|
|
|
static const struct frame_unwind *
|
2003-07-17 06:29:13 +08:00
|
|
|
d10v_frame_sniffer (struct frame_info *next_frame)
|
2003-01-30 23:11:20 +08:00
|
|
|
{
|
|
|
|
return &d10v_frame_unwind;
|
|
|
|
}
|
|
|
|
|
2003-04-02 03:55:03 +08:00
|
|
|
static CORE_ADDR
|
|
|
|
d10v_frame_base_address (struct frame_info *next_frame, void **this_cache)
|
|
|
|
{
|
|
|
|
struct d10v_unwind_cache *info
|
|
|
|
= d10v_frame_unwind_cache (next_frame, this_cache);
|
|
|
|
return info->base;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct frame_base d10v_frame_base = {
|
|
|
|
&d10v_frame_unwind,
|
|
|
|
d10v_frame_base_address,
|
|
|
|
d10v_frame_base_address,
|
|
|
|
d10v_frame_base_address
|
|
|
|
};
|
|
|
|
|
2003-03-06 07:14:18 +08:00
|
|
|
/* Assuming NEXT_FRAME->prev is a dummy, return the frame ID of that
|
|
|
|
dummy frame. The frame ID's base needs to match the TOS value
|
|
|
|
saved by save_dummy_frame_tos(), and the PC match the dummy frame's
|
|
|
|
breakpoint. */
|
|
|
|
|
|
|
|
static struct frame_id
|
|
|
|
d10v_unwind_dummy_id (struct gdbarch *gdbarch, struct frame_info *next_frame)
|
|
|
|
{
|
2003-06-14 04:37:28 +08:00
|
|
|
return frame_id_build (d10v_unwind_sp (gdbarch, next_frame),
|
|
|
|
frame_pc_unwind (next_frame));
|
2003-03-06 07:14:18 +08:00
|
|
|
}
|
|
|
|
|
1999-06-15 02:08:47 +08:00
|
|
|
static gdbarch_init_ftype d10v_gdbarch_init;
|
1999-11-17 10:31:06 +08:00
|
|
|
|
1999-06-15 02:08:47 +08:00
|
|
|
static struct gdbarch *
|
2000-07-30 09:48:28 +08:00
|
|
|
d10v_gdbarch_init (struct gdbarch_info info, struct gdbarch_list *arches)
|
1999-06-15 02:08:47 +08:00
|
|
|
{
|
|
|
|
struct gdbarch *gdbarch;
|
1999-11-17 10:31:06 +08:00
|
|
|
int d10v_num_regs;
|
|
|
|
struct gdbarch_tdep *tdep;
|
|
|
|
gdbarch_register_name_ftype *d10v_register_name;
|
2000-08-02 19:05:50 +08:00
|
|
|
gdbarch_register_sim_regno_ftype *d10v_register_sim_regno;
|
1999-06-15 02:08:47 +08:00
|
|
|
|
2003-10-10 08:25:43 +08:00
|
|
|
/* Find a candidate among the list of pre-declared architectures. */
|
1999-11-17 10:31:06 +08:00
|
|
|
arches = gdbarch_list_lookup_by_info (arches, &info);
|
1999-06-15 02:08:47 +08:00
|
|
|
if (arches != NULL)
|
|
|
|
return arches->gdbarch;
|
1999-11-17 10:31:06 +08:00
|
|
|
|
|
|
|
/* None found, create a new architecture from the information
|
2003-10-10 08:25:43 +08:00
|
|
|
provided. */
|
1999-11-17 10:31:06 +08:00
|
|
|
tdep = XMALLOC (struct gdbarch_tdep);
|
|
|
|
gdbarch = gdbarch_alloc (&info, tdep);
|
|
|
|
|
|
|
|
switch (info.bfd_arch_info->mach)
|
|
|
|
{
|
|
|
|
case bfd_mach_d10v_ts2:
|
|
|
|
d10v_num_regs = 37;
|
|
|
|
d10v_register_name = d10v_ts2_register_name;
|
2000-08-02 19:05:50 +08:00
|
|
|
d10v_register_sim_regno = d10v_ts2_register_sim_regno;
|
1999-11-17 10:31:06 +08:00
|
|
|
tdep->a0_regnum = TS2_A0_REGNUM;
|
|
|
|
tdep->nr_dmap_regs = TS2_NR_DMAP_REGS;
|
|
|
|
tdep->dmap_register = d10v_ts2_dmap_register;
|
|
|
|
tdep->imap_register = d10v_ts2_imap_register;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
case bfd_mach_d10v_ts3:
|
|
|
|
d10v_num_regs = 42;
|
|
|
|
d10v_register_name = d10v_ts3_register_name;
|
2000-08-02 19:05:50 +08:00
|
|
|
d10v_register_sim_regno = d10v_ts3_register_sim_regno;
|
1999-11-17 10:31:06 +08:00
|
|
|
tdep->a0_regnum = TS3_A0_REGNUM;
|
|
|
|
tdep->nr_dmap_regs = TS3_NR_DMAP_REGS;
|
|
|
|
tdep->dmap_register = d10v_ts3_dmap_register;
|
|
|
|
tdep->imap_register = d10v_ts3_imap_register;
|
|
|
|
break;
|
|
|
|
}
|
1999-06-15 02:08:47 +08:00
|
|
|
|
|
|
|
set_gdbarch_read_pc (gdbarch, d10v_read_pc);
|
|
|
|
set_gdbarch_write_pc (gdbarch, d10v_write_pc);
|
2003-06-09 09:02:07 +08:00
|
|
|
set_gdbarch_unwind_sp (gdbarch, d10v_unwind_sp);
|
1999-06-15 02:08:47 +08:00
|
|
|
|
|
|
|
set_gdbarch_num_regs (gdbarch, d10v_num_regs);
|
2003-04-26 05:20:58 +08:00
|
|
|
set_gdbarch_sp_regnum (gdbarch, D10V_SP_REGNUM);
|
1999-06-15 02:08:47 +08:00
|
|
|
set_gdbarch_register_name (gdbarch, d10v_register_name);
|
2003-03-02 01:59:12 +08:00
|
|
|
set_gdbarch_register_type (gdbarch, d10v_register_type);
|
1999-06-15 02:08:47 +08:00
|
|
|
|
Clean up the D10V port so that GDB and the target program no
longer disagree on how big pointers are.
* findvar.c (value_from_register): Remove special case code for D10V.
* printcmd.c (print_frame_args): Same.
* valops.c (value_at, value_fetch_lazy): Same.
* values.c (unpack_long): Same.
* gdbarch.sh: Changes to effect the following:
* gdbarch.h (GDB_TARGET_IS_D10V, D10V_MAKE_DADDR,
gdbarch_d10v_make_daddr_ftype, gdbarch_d10v_make_daddr,
set_gdbarch_d10v_make_daddr, D10V_MAKE_IADDR,
gdbarch_d10v_make_iaddr_ftype, gdbarch_d10v_make_iaddr,
set_gdbarch_d10v_make_iaddr, D10V_DADDR_P,
gdbarch_d10v_daddr_p_ftype, gdbarch_d10v_daddr_p,
set_gdbarch_d10v_daddr_p, D10V_IADDR_P,
gdbarch_d10v_iaddr_p_ftype, gdbarch_d10v_iaddr_p,
set_gdbarch_d10v_iaddr_p, D10V_CONVERT_DADDR_TO_RAW,
gdbarch_d10v_convert_daddr_to_raw_ftype,
gdbarch_d10v_convert_daddr_to_raw,
set_gdbarch_d10v_convert_daddr_to_raw, D10V_CONVERT_IADDR_TO_RAW,
gdbarch_d10v_convert_iaddr_to_raw_ftype,
gdbarch_d10v_convert_iaddr_to_raw,
set_gdbarch_d10v_convert_iaddr_to_raw): Delete declarations.
* gdbarch.c: Delete the corresponding definitions.
(struct gdbarch): Delete members d10v_make_daddr,
d10v_make_iaddr, d10v_daddr_p, d10v_iaddr_p,
d10v_convert_daddr_to_raw, and d10v_convert_iaddr_to_raw.
(startup_gdbarch): Remove initializers for the above.
(verify_gdbarch, gdbarch_dump): Don't verify or dump them any
more.
* d10v-tdep.c (d10v_register_virtual_type): Rather that
claiming the stack pointer and PC are 32 bits long (which they
aren't), say that the stack pointer is an int16_t, and the
program counter is a function pointer. This allows the rest
of GDB to make the appropriate conversions between the code
pointer format and real addresses.
(d10v_register_convertible, d10v_register_convert_to_virtual,
d10v_register_convert_to_raw): Delete function; no registers
are convertible now, so we use
generic_register_convertible_not instead.
(d10v_address_to_pointer, d10v_pointer_to_address): New gdbarch
methods.
(d10v_push_arguments, d10v_extract_return_value): Remove special
cases for code and data pointers.
(d10v_gdbarch_init): Set gdbarch_ptr_bit to 16, so that GDB and
the target agree on how large pointers are. Say that addresses
are 32 bits long. Register the address_to_pointer and
pointer_to_address conversion functions. Since no registers are
convertible now, register generic_register_convertible_not as the
gdbarch_register_convertible method instead of
d10v_register_convertible. Remove registrations for
d10v_register_convert_to_virtual,
d10v_register_convert_to_raw, gdbarch_d10v_make_daddr,
gdbarch_d10v_make_iaddr, gdbarch_d10v_daddr_p,
gdbarch_d10v_iaddr_p, gdbarch_d10v_convert_daddr_to_raw, and
gdbarch_d10v_convert_iaddr_to_raw.
2001-07-11 05:24:48 +08:00
|
|
|
set_gdbarch_ptr_bit (gdbarch, 2 * TARGET_CHAR_BIT);
|
|
|
|
set_gdbarch_addr_bit (gdbarch, 32);
|
|
|
|
set_gdbarch_address_to_pointer (gdbarch, d10v_address_to_pointer);
|
|
|
|
set_gdbarch_pointer_to_address (gdbarch, d10v_pointer_to_address);
|
2001-10-16 02:18:30 +08:00
|
|
|
set_gdbarch_integer_to_address (gdbarch, d10v_integer_to_address);
|
1999-06-15 02:08:47 +08:00
|
|
|
set_gdbarch_short_bit (gdbarch, 2 * TARGET_CHAR_BIT);
|
|
|
|
set_gdbarch_int_bit (gdbarch, 2 * TARGET_CHAR_BIT);
|
|
|
|
set_gdbarch_long_bit (gdbarch, 4 * TARGET_CHAR_BIT);
|
2001-12-05 10:05:04 +08:00
|
|
|
set_gdbarch_long_long_bit (gdbarch, 8 * TARGET_CHAR_BIT);
|
2000-06-02 09:59:13 +08:00
|
|
|
/* NOTE: The d10v as a 32 bit ``float'' and ``double''. ``long
|
2003-10-10 08:25:43 +08:00
|
|
|
double'' is 64 bits. */
|
1999-06-15 02:08:47 +08:00
|
|
|
set_gdbarch_float_bit (gdbarch, 4 * TARGET_CHAR_BIT);
|
|
|
|
set_gdbarch_double_bit (gdbarch, 4 * TARGET_CHAR_BIT);
|
|
|
|
set_gdbarch_long_double_bit (gdbarch, 8 * TARGET_CHAR_BIT);
|
2000-06-02 09:59:13 +08:00
|
|
|
switch (info.byte_order)
|
|
|
|
{
|
2002-01-05 12:30:46 +08:00
|
|
|
case BFD_ENDIAN_BIG:
|
2000-06-02 09:59:13 +08:00
|
|
|
set_gdbarch_float_format (gdbarch, &floatformat_ieee_single_big);
|
|
|
|
set_gdbarch_double_format (gdbarch, &floatformat_ieee_single_big);
|
|
|
|
set_gdbarch_long_double_format (gdbarch, &floatformat_ieee_double_big);
|
|
|
|
break;
|
2001-12-16 00:53:24 +08:00
|
|
|
case BFD_ENDIAN_LITTLE:
|
2000-06-02 09:59:13 +08:00
|
|
|
set_gdbarch_float_format (gdbarch, &floatformat_ieee_single_little);
|
|
|
|
set_gdbarch_double_format (gdbarch, &floatformat_ieee_single_little);
|
2003-10-10 08:25:43 +08:00
|
|
|
set_gdbarch_long_double_format (gdbarch,
|
|
|
|
&floatformat_ieee_double_little);
|
2000-06-02 09:59:13 +08:00
|
|
|
break;
|
|
|
|
default:
|
2001-02-08 14:03:54 +08:00
|
|
|
internal_error (__FILE__, __LINE__,
|
|
|
|
"d10v_gdbarch_init: bad byte order for float format");
|
2000-06-02 09:59:13 +08:00
|
|
|
}
|
1999-06-15 02:08:47 +08:00
|
|
|
|
2003-11-09 22:20:55 +08:00
|
|
|
set_gdbarch_return_value (gdbarch, d10v_return_value);
|
2003-05-04 03:39:23 +08:00
|
|
|
set_gdbarch_push_dummy_code (gdbarch, d10v_push_dummy_code);
|
2003-03-27 23:09:48 +08:00
|
|
|
set_gdbarch_push_dummy_call (gdbarch, d10v_push_dummy_call);
|
1999-06-15 02:08:47 +08:00
|
|
|
|
|
|
|
set_gdbarch_skip_prologue (gdbarch, d10v_skip_prologue);
|
|
|
|
set_gdbarch_inner_than (gdbarch, core_addr_lessthan);
|
|
|
|
set_gdbarch_decr_pc_after_break (gdbarch, 4);
|
|
|
|
set_gdbarch_breakpoint_from_pc (gdbarch, d10v_breakpoint_from_pc);
|
|
|
|
|
2003-10-10 08:25:43 +08:00
|
|
|
set_gdbarch_remote_translate_xfer_address (gdbarch,
|
|
|
|
remote_d10v_translate_xfer_address);
|
1999-06-15 02:08:47 +08:00
|
|
|
|
|
|
|
set_gdbarch_frame_args_skip (gdbarch, 0);
|
2003-10-10 08:25:43 +08:00
|
|
|
set_gdbarch_frameless_function_invocation (gdbarch,
|
|
|
|
frameless_look_for_prologue);
|
2002-11-23 09:39:25 +08:00
|
|
|
|
2003-05-01 23:37:45 +08:00
|
|
|
set_gdbarch_frame_align (gdbarch, d10v_frame_align);
|
1999-06-15 02:08:47 +08:00
|
|
|
|
2000-08-02 19:05:50 +08:00
|
|
|
set_gdbarch_register_sim_regno (gdbarch, d10v_register_sim_regno);
|
|
|
|
|
2003-02-03 02:53:22 +08:00
|
|
|
set_gdbarch_print_registers_info (gdbarch, d10v_print_registers_info);
|
|
|
|
|
2003-07-17 06:29:13 +08:00
|
|
|
frame_unwind_append_sniffer (gdbarch, d10v_frame_sniffer);
|
2003-04-02 03:55:03 +08:00
|
|
|
frame_base_set_default (gdbarch, &d10v_frame_base);
|
2003-01-30 23:11:20 +08:00
|
|
|
|
2003-06-14 04:37:28 +08:00
|
|
|
/* Methods for saving / extracting a dummy frame's ID. The ID's
|
|
|
|
stack address must match the SP value returned by
|
|
|
|
PUSH_DUMMY_CALL, and saved by generic_save_dummy_frame_tos. */
|
2003-03-06 07:14:18 +08:00
|
|
|
set_gdbarch_unwind_dummy_id (gdbarch, d10v_unwind_dummy_id);
|
|
|
|
|
2003-03-10 23:28:41 +08:00
|
|
|
/* Return the unwound PC value. */
|
|
|
|
set_gdbarch_unwind_pc (gdbarch, d10v_unwind_pc);
|
|
|
|
|
2003-04-26 06:14:05 +08:00
|
|
|
set_gdbarch_print_insn (gdbarch, print_insn_d10v);
|
|
|
|
|
1999-06-15 02:08:47 +08:00
|
|
|
return gdbarch;
|
|
|
|
}
|
|
|
|
|
1999-04-16 09:35:26 +08:00
|
|
|
void
|
2000-07-30 09:48:28 +08:00
|
|
|
_initialize_d10v_tdep (void)
|
1999-04-16 09:35:26 +08:00
|
|
|
{
|
1999-06-15 02:08:47 +08:00
|
|
|
register_gdbarch_init (bfd_arch_d10v, d10v_gdbarch_init);
|
|
|
|
|
1999-04-16 09:35:26 +08:00
|
|
|
target_resume_hook = d10v_eva_prepare_to_trace;
|
|
|
|
target_wait_loop_hook = d10v_eva_get_trace_data;
|
|
|
|
|
2003-10-10 08:25:43 +08:00
|
|
|
deprecate_cmd (add_com ("regs", class_vars, show_regs,
|
|
|
|
"Print all registers"),
|
2003-02-03 02:53:22 +08:00
|
|
|
"info registers");
|
1999-04-16 09:35:26 +08:00
|
|
|
|
1999-09-14 05:40:00 +08:00
|
|
|
add_com ("itrace", class_support, trace_command,
|
1999-04-16 09:35:26 +08:00
|
|
|
"Enable tracing of instruction execution.");
|
|
|
|
|
1999-09-14 05:40:00 +08:00
|
|
|
add_com ("iuntrace", class_support, untrace_command,
|
1999-04-16 09:35:26 +08:00
|
|
|
"Disable tracing of instruction execution.");
|
|
|
|
|
1999-09-14 05:40:00 +08:00
|
|
|
add_com ("itdisassemble", class_vars, tdisassemble_command,
|
1999-04-16 09:35:26 +08:00
|
|
|
"Disassemble the trace buffer.\n\
|
|
|
|
Two optional arguments specify a range of trace buffer entries\n\
|
|
|
|
as reported by info trace (NOT addresses!).");
|
|
|
|
|
1999-09-14 05:40:00 +08:00
|
|
|
add_info ("itrace", trace_info,
|
1999-04-16 09:35:26 +08:00
|
|
|
"Display info about the trace data buffer.");
|
|
|
|
|
2003-04-26 05:20:58 +08:00
|
|
|
add_setshow_boolean_cmd ("itracedisplay", no_class, &trace_display,
|
|
|
|
"Set automatic display of trace.\n",
|
|
|
|
"Show automatic display of trace.\n",
|
|
|
|
NULL, NULL, &setlist, &showlist);
|
|
|
|
add_setshow_boolean_cmd ("itracesource", no_class,
|
|
|
|
&default_trace_show_source,
|
|
|
|
"Set display of source code with trace.\n",
|
|
|
|
"Show display of source code with trace.\n",
|
|
|
|
NULL, NULL, &setlist, &showlist);
|
1999-07-08 04:19:36 +08:00
|
|
|
}
|