1999-04-16 03:35:26 +02:00
|
|
|
/* Remote target glue for the ROM68K ROM monitor.
|
2001-03-06 09:22:02 +01:00
|
|
|
Copyright 1988, 1991, 1992, 1993, 1994, 1995, 1998, 1999, 2000, 2001
|
|
|
|
Free Software Foundation, Inc.
|
1999-04-16 03:35:26 +02:00
|
|
|
|
1999-07-07 22:19:36 +02:00
|
|
|
This file is part of GDB.
|
1999-04-16 03:35:26 +02:00
|
|
|
|
1999-07-07 22:19:36 +02: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 03:35:26 +02:00
|
|
|
|
1999-07-07 22:19:36 +02: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 03:35:26 +02:00
|
|
|
|
1999-07-07 22:19:36 +02: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 03:35:26 +02:00
|
|
|
|
|
|
|
#include "defs.h"
|
|
|
|
#include "gdbcore.h"
|
|
|
|
#include "target.h"
|
|
|
|
#include "monitor.h"
|
|
|
|
#include "serial.h"
|
2001-03-01 02:39:22 +01:00
|
|
|
#include "regcache.h"
|
2001-09-07 23:27:36 +02:00
|
|
|
#include "value.h"
|
1999-04-16 03:35:26 +02:00
|
|
|
|
2002-11-23 20:45:12 +01:00
|
|
|
#include "m68k-tdep.h"
|
|
|
|
|
2000-05-28 03:12:42 +02:00
|
|
|
static void rom68k_open (char *args, int from_tty);
|
1999-04-16 03:35:26 +02:00
|
|
|
|
2001-09-07 23:27:36 +02:00
|
|
|
/* Return true if C is a hex digit.
|
|
|
|
We can't use isxdigit here: that is affected by the current locale;
|
|
|
|
ROM68K is not. */
|
|
|
|
static int
|
|
|
|
is_hex_digit (int c)
|
|
|
|
{
|
|
|
|
return (('0' <= c && c <= '9')
|
|
|
|
|| ('a' <= c && c <= 'f')
|
|
|
|
|| ('A' <= c && c <= 'F'));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Convert hex digit A to a number. */
|
|
|
|
static int
|
|
|
|
hex_digit_value (int a)
|
|
|
|
{
|
|
|
|
if (a >= '0' && a <= '9')
|
|
|
|
return a - '0';
|
|
|
|
else if (a >= 'a' && a <= 'f')
|
|
|
|
return a - 'a' + 10;
|
|
|
|
else if (a >= 'A' && a <= 'F')
|
|
|
|
return a - 'A' + 10;
|
|
|
|
else
|
|
|
|
error ("Invalid hex digit %d", a);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Return true iff C is a whitespace character.
|
|
|
|
We can't use isspace here: that is affected by the current locale;
|
|
|
|
ROM68K is not. */
|
|
|
|
static int
|
|
|
|
is_whitespace (int c)
|
|
|
|
{
|
|
|
|
return (c == ' '
|
|
|
|
|| c == '\r'
|
|
|
|
|| c == '\n'
|
|
|
|
|| c == '\t'
|
|
|
|
|| c == '\f');
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Parse a string of hex digits starting at HEX, supply them as the
|
|
|
|
value of register REGNO, skip any whitespace, and return a pointer
|
|
|
|
to the next character.
|
|
|
|
|
|
|
|
There is a function in monitor.c, monitor_supply_register, which is
|
|
|
|
supposed to do this job. However, there is some rather odd stuff
|
|
|
|
in there (whitespace characters don't terminate numbers, for
|
|
|
|
example) that is incorrect for ROM68k. It's basically impossible
|
|
|
|
to safely tweak monitor_supply_register --- it's used by a zillion
|
|
|
|
other monitors; who knows what behaviors they're depending on. So
|
|
|
|
instead, we'll just use our own function, which can behave exactly
|
|
|
|
the way we want it to. */
|
|
|
|
static char *
|
|
|
|
rom68k_supply_one_register (int regno, unsigned char *hex)
|
|
|
|
{
|
|
|
|
ULONGEST value;
|
2003-05-08 Andrew Cagney <cagney@redhat.com>
* regcache.h (max_register_size): Delete declaration.
* regcache.c (max_register_size): Delete function.
(struct regcache_descr): Delete field "max_register_size".
(init_regcache_descr, init_legacy_regcache_descr): Assert that all
registers fit in MAX_REGISTER_SIZE.
(regcache_save): Replace max_register_size with MAX_REGISTER_SIZE.
(regcache_restore, regcache_xfer_part, regcache_dump): Ditto.
* thread-db.c: Replace max_register_size with MAX_REGISTER_SIZE.
* sh-tdep.c, rom68k-rom.c, remote-sim.c, remote-mips.c: Ditto.
* remote-e7000.c, monitor.c, mipsv4-nat.c, mips-nat.c: Ditto.
* m68klinux-nat.c, lynx-nat.c, irix4-nat.c: Ditto.
* hpux-thread.c, hppah-nat.c, hppab-nat.c, hppa-tdep.c: Ditto.
* dve3900-rom.c, hppa-tdep.c: Ditto.
2003-05-09 00:33:14 +02:00
|
|
|
unsigned char regbuf[MAX_REGISTER_SIZE];
|
2001-09-07 23:27:36 +02:00
|
|
|
|
|
|
|
value = 0;
|
|
|
|
while (*hex != '\0')
|
|
|
|
if (is_hex_digit (*hex))
|
|
|
|
value = (value * 16) + hex_digit_value (*hex++);
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* Skip any whitespace. */
|
|
|
|
while (is_whitespace (*hex))
|
|
|
|
hex++;
|
|
|
|
|
2004-08-02 Andrew Cagney <cagney@gnu.org>
Replace DEPRECATED_REGISTER_RAW_SIZE with register_size.
* rs6000-tdep.c (rs6000_push_dummy_call)
(rs6000_extract_return_value): Use register_size.
* xstormy16-tdep.c (xstormy16_get_saved_register)
(xstormy16_extract_return_value): Ditto.
* valops.c (value_assign): Ditto.
* v850ice.c (v850ice_fetch_registers, v850ice_store_registers):
* v850-tdep.c (v850_extract_return_value): Ditto.
* tracepoint.c (collect_symbol): Ditto.
* target.c (debug_print_register): Ditto.
* stack.c (frame_info): Ditto.
* rs6000-nat.c (ARCH64, fetch_register, store_register): Ditto.
* rom68k-rom.c (rom68k_supply_one_register): Ditto.
* remote.c (struct packet_reg, remote_wait, remote_async_wait)
(store_register_using_P): Ditto.
* remote-vxmips.c (vx_read_register, vx_write_register): Ditto.
* remote-sim.c (gdbsim_fetch_register, gdbsim_store_register): Ditto.
* remote-mips.c (mips_wait, mips_fetch_registers): Ditto.
* remote-e7000.c (fetch_regs_from_dump, sub2_from_pc): Ditto.
* regcache.c (deprecated_read_register_bytes)
(deprecated_write_register_bytes, read_register)
(write_register): Ditto.
* ppc-linux-nat.c (fetch_altivec_register, fetch_register)
(supply_vrregset, store_altivec_register, fill_vrregset): Ditto.
* monitor.c (monitor_supply_register, monitor_fetch_register)
(monitor_store_register): Ditto.
* mn10300-tdep.c (mn10300_pop_frame_regular)
(mn10300_print_register): Ditto.
* mipsv4-nat.c (fill_fpregset): Ditto.
* mips-linux-tdep.c (supply_32bit_reg, fill_fpregset)
(mips64_fill_fpregset): Ditto.
* mi/mi-main.c (register_changed_p, get_register)
(mi_cmd_data_write_register_values): Ditto.
* lynx-nat.c (fetch_inferior_registers, store_inferior_registers):
* irix5-nat.c (fill_gregset, fetch_core_registers):
* infrun.c (write_inferior_status_register): Ditto.
* infptrace.c (fetch_register, store_register): Ditto.
* infcmd.c (default_print_registers_info): Ditto.
* ia64-linux-nat.c (COPY_REG, fill_fpregset): Ditto.
* ia64-aix-nat.c (COPY_REG, fill_gregset): Ditto.
* i386gnu-nat.c (gnu_store_registers, fill): Ditto.
* hpux-thread.c (hpux_thread_fetch_registers)
(hpux_thread_store_registers): Ditto.
* hppah-nat.c (store_inferior_registers, fetch_register):
* findvar.c (value_from_register): Ditto.
* dve3900-rom.c (fetch_bitmapped_register):
* cris-tdep.c (cris_gdbarch_init): Ditto.
* alpha-tdep.h: Ditto.
* aix-thread.c (pd_enable, fill_sprs64, fill_sprs32): Ditto.
2004-08-03 02:57:27 +02:00
|
|
|
store_unsigned_integer (regbuf, register_size (current_gdbarch, regno), value);
|
2004-07-21 Andrew Cagney <cagney@gnu.org>
Use regcache_raw_supply instead of supply_register.
* regcache.h (supply_register): Delete declaration.
* regcache.c (supply_register): Delete function.
* wince.c (do_child_fetch_inferior_registers): Update.
* win32-nat.c (do_child_fetch_inferior_registers)
(fetch_elf_core_registers): Update.
* v850ice.c (v850ice_fetch_registers): Update.
* thread-db.c (thread_db_store_registers): Update.
* sol-thread.c (sol_thread_store_registers): Update.
* shnbsd-tdep.c (shnbsd_supply_reg): Update.
* rs6000-nat.c (fetch_register): Update.
* rom68k-rom.c (rom68k_supply_one_register): Update.
* remote.c (remote_wait, remote_async_wait): Update.
* remote-st.c (get_hex_regs): Update.
* remote-sim.c (gdbsim_fetch_register): Update.
* remote-sds.c (sds_fetch_registers): Update.
* remote-rdp.c (remote_rdp_fetch_register): Update.
* remote-rdi.c (arm_rdi_fetch_registers): Update.
* remote-mips.c (mips_wait, mips_fetch_registers): Update.
* remote-m32r-sdi.c (m32r_fetch_register): Update.
* remote-hms.c (init_hms_cmds): Update.
* remote-est.c (init_est_cmds): Update.
* remote-e7000.c (get_hex_regs, fetch_regs_from_dump)
(e7000_fetch_registers, sub2_from_pc, e7000_wait): Update.
* ppcnbsd-tdep.c (ppcnbsd_supply_reg, ppcnbsd_supply_fpreg): Update.
* ppc-linux-nat.c (fetch_altivec_register, fetch_spe_register)
(fetch_register, supply_vrregset, supply_vrregset)
(fetch_spe_registers): Update.
* ppc-bdm.c (bdm_ppc_fetch_registers): Update.
* monitor.c (monitor_supply_register): Update.
* mipsv4-nat.c (supply_gregset, supply_fpregset): Update.
* mipsnbsd-tdep.c (mipsnbsd_supply_reg)
(mipsnbsd_supply_fpreg): Update.
* mips-nat.c (fetch_inferior_registers)
(fetch_core_registers): Update.
* mips-linux-tdep.c (supply_32bit_reg, supply_gregset)
(supply_fpregset, mips64_supply_gregset)
(mips64_supply_fpregset): Update.
* m68klinux-nat.c (fetch_register, supply_gregset)
(supply_fpregset): Update.
* m68k-tdep.c (supply_gregset, supply_fpregset): Update.
* m32r-rom.c (init_m32r_cmds, init_mon2000_cmds): Update.
* lynx-nat.c (fetch_inferior_registers, fetch_core_registers): Update.
* irix5-nat.c (supply_gregset, supply_fpregset): Update.
* infptrace.c (fetch_register): Update.
* ia64-linux-nat.c (supply_gregset, supply_fpregset): Update.
* ia64-aix-nat.c (supply_gregset, supply_fpregset): Update.
* i386gnu-nat.c (fetch_fpregs, supply_gregset)
(gnu_fetch_registers, gnu_store_registers): Update.
* i386-nto-tdep.c (i386nto_supply_gregset): Update.
* i386-linux-nat.c (fetch_register, supply_gregset)
(dummy_sse_values): Update.
* hpux-thread.c (hpux_thread_fetch_registers): Update.
* hppah-nat.c (fetch_register): Update.
* hppa-linux-nat.c (fetch_register, supply_gregset)
(supply_fpregset): Update.
* go32-nat.c (fetch_register): Update.
* dve3900-rom.c (fetch_bitmapped_register)
(_initialize_r3900_rom): Update.
* cris-tdep.c (supply_gregset): Update.
* abug-rom.c (init_abug_cmds): Update.
* core-aout.c (fetch_core_registers): Update.
* armnbsd-nat.c (supply_gregset, supply_fparegset)
(fetch_register, fetch_fp_register): Update.
* arm-linux-nat.c (fetch_nwfpe_single, fetch_nwfpe_none)
(fetch_nwfpe_extended, fetch_fpregister, fetch_fpregs)
(fetch_register, fetch_regs, supply_gregset, supply_fpregset): Update.
* alphanbsd-tdep.c (fetch_core_registers): Update.
* alpha-tdep.c (alpha_supply_int_regs, alpha_supply_fp_regs): Update.
* alpha-nat.c (fetch_osf_core_registers)
(fetch_osf_core_registers, fetch_osf_core_registers): Update.
* aix-thread.c (supply_gprs64, supply_reg32, supply_fprs)
(supply_sprs64, supply_sprs32, fetch_regs_kernel_thread): Update.
2004-07-22 03:31:49 +02:00
|
|
|
regcache_raw_supply (current_regcache, regno, regbuf);
|
2001-09-07 23:27:36 +02:00
|
|
|
|
|
|
|
return hex;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1999-04-16 03:35:26 +02:00
|
|
|
static void
|
2000-07-30 03:48:28 +02:00
|
|
|
rom68k_supply_register (char *regname, int regnamelen, char *val, int vallen)
|
1999-04-16 03:35:26 +02:00
|
|
|
{
|
|
|
|
int numregs;
|
|
|
|
int regno;
|
|
|
|
|
|
|
|
numregs = 1;
|
|
|
|
regno = -1;
|
|
|
|
|
|
|
|
if (regnamelen == 2)
|
|
|
|
switch (regname[0])
|
|
|
|
{
|
|
|
|
case 'S':
|
|
|
|
if (regname[1] == 'R')
|
|
|
|
regno = PS_REGNUM;
|
|
|
|
break;
|
|
|
|
case 'P':
|
|
|
|
if (regname[1] == 'C')
|
|
|
|
regno = PC_REGNUM;
|
|
|
|
break;
|
|
|
|
case 'D':
|
|
|
|
if (regname[1] != 'R')
|
|
|
|
break;
|
2002-11-23 20:45:12 +01:00
|
|
|
regno = M68K_D0_REGNUM;
|
1999-04-16 03:35:26 +02:00
|
|
|
numregs = 8;
|
|
|
|
break;
|
|
|
|
case 'A':
|
|
|
|
if (regname[1] != 'R')
|
|
|
|
break;
|
2002-11-23 20:45:12 +01:00
|
|
|
regno = M68K_A0_REGNUM;
|
1999-04-16 03:35:26 +02:00
|
|
|
numregs = 7;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else if (regnamelen == 3)
|
|
|
|
switch (regname[0])
|
|
|
|
{
|
|
|
|
case 'I':
|
|
|
|
if (regname[1] == 'S' && regname[2] == 'P')
|
|
|
|
regno = SP_REGNUM;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (regno >= 0)
|
|
|
|
while (numregs-- > 0)
|
2001-09-07 23:27:36 +02:00
|
|
|
val = rom68k_supply_one_register (regno++, val);
|
1999-04-16 03:35:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* This array of registers need to match the indexes used by GDB.
|
|
|
|
This exists because the various ROM monitors use different strings
|
|
|
|
than does GDB, and don't necessarily support all the registers
|
|
|
|
either. So, typing "info reg sp" becomes a "r30". */
|
|
|
|
|
2002-06-26 17:14:32 +02:00
|
|
|
static const char *
|
|
|
|
rom68k_regname (int index)
|
1999-07-07 22:19:36 +02:00
|
|
|
{
|
2002-06-26 17:14:32 +02:00
|
|
|
|
|
|
|
static char *regnames[] =
|
|
|
|
{
|
|
|
|
"D0", "D1", "D2", "D3", "D4", "D5", "D6", "D7",
|
|
|
|
"A0", "A1", "A2", "A3", "A4", "A5", "A6", "ISP",
|
|
|
|
"SR", "PC"
|
|
|
|
};
|
|
|
|
|
|
|
|
if ((index >= (sizeof (regnames) / sizeof(regnames[0])))
|
|
|
|
|| (index < 0) || (index >= NUM_REGS))
|
|
|
|
return NULL;
|
|
|
|
else
|
|
|
|
return regnames[index];
|
|
|
|
|
|
|
|
}
|
1999-04-16 03:35:26 +02:00
|
|
|
|
|
|
|
/* Define the monitor command strings. Since these are passed directly
|
|
|
|
through to a printf style function, we may include formatting
|
|
|
|
strings. We also need a CR or LF on the end. */
|
|
|
|
|
|
|
|
static struct target_ops rom68k_ops;
|
|
|
|
|
1999-07-07 22:19:36 +02:00
|
|
|
static char *rom68k_inits[] =
|
|
|
|
{".\r\r", NULL}; /* Exits pm/pr & download cmds */
|
1999-04-16 03:35:26 +02:00
|
|
|
|
1999-07-07 22:19:36 +02:00
|
|
|
static struct monitor_ops rom68k_cmds;
|
1999-04-16 03:35:26 +02:00
|
|
|
|
1999-07-07 22:19:36 +02:00
|
|
|
static void
|
1999-04-26 20:34:20 +02:00
|
|
|
init_rom68k_cmds (void)
|
1999-04-16 03:35:26 +02:00
|
|
|
{
|
2001-09-11 01:54:16 +02:00
|
|
|
rom68k_cmds.flags = MO_PRINT_PROGRAM_OUTPUT;
|
1999-07-07 22:19:36 +02:00
|
|
|
rom68k_cmds.init = rom68k_inits; /* monitor init string */
|
|
|
|
rom68k_cmds.cont = "go\r";
|
|
|
|
rom68k_cmds.step = "st\r";
|
|
|
|
rom68k_cmds.stop = NULL;
|
|
|
|
rom68k_cmds.set_break = "db %x\r";
|
|
|
|
rom68k_cmds.clr_break = "cb %x\r";
|
|
|
|
rom68k_cmds.clr_all_break = "cb *\r";
|
|
|
|
rom68k_cmds.fill = "fm %x %x %x\r";
|
|
|
|
rom68k_cmds.setmem.cmdb = "pm %x %x\r";
|
|
|
|
rom68k_cmds.setmem.cmdw = "pm.w %x %x\r";
|
|
|
|
rom68k_cmds.setmem.cmdl = "pm.l %x %x\r";
|
|
|
|
rom68k_cmds.setmem.cmdll = NULL;
|
|
|
|
rom68k_cmds.setmem.resp_delim = NULL;
|
|
|
|
rom68k_cmds.setmem.term = NULL;
|
|
|
|
rom68k_cmds.setmem.term_cmd = NULL;
|
|
|
|
rom68k_cmds.getmem.cmdb = "dm %x %x\r";
|
|
|
|
rom68k_cmds.getmem.cmdw = "dm.w %x %x\r";
|
|
|
|
rom68k_cmds.getmem.cmdl = "dm.l %x %x\r";
|
|
|
|
rom68k_cmds.getmem.cmdll = NULL;
|
|
|
|
rom68k_cmds.getmem.resp_delim = " ";
|
|
|
|
rom68k_cmds.getmem.term = NULL;
|
|
|
|
rom68k_cmds.getmem.term_cmd = NULL;
|
|
|
|
rom68k_cmds.setreg.cmd = "pr %s %x\r";
|
|
|
|
rom68k_cmds.setreg.resp_delim = NULL;
|
|
|
|
rom68k_cmds.setreg.term = NULL;
|
|
|
|
rom68k_cmds.setreg.term_cmd = NULL;
|
|
|
|
rom68k_cmds.getreg.cmd = "pr %s\r";
|
|
|
|
rom68k_cmds.getreg.resp_delim = ": ";
|
|
|
|
rom68k_cmds.getreg.term = "= ";
|
|
|
|
rom68k_cmds.getreg.term_cmd = ".\r";
|
|
|
|
rom68k_cmds.dump_registers = "dr\r";
|
1999-04-16 03:35:26 +02:00
|
|
|
rom68k_cmds.register_pattern =
|
1999-04-26 20:34:20 +02:00
|
|
|
"\\(\\w+\\)=\\([0-9a-fA-F]+\\( +[0-9a-fA-F]+\\b\\)*\\)";
|
1999-07-07 22:19:36 +02:00
|
|
|
rom68k_cmds.supply_register = rom68k_supply_register;
|
|
|
|
rom68k_cmds.load_routine = NULL;
|
|
|
|
rom68k_cmds.load = "dc\r";
|
|
|
|
rom68k_cmds.loadresp = "Waiting for S-records from host... ";
|
|
|
|
rom68k_cmds.prompt = "ROM68K :-> ";
|
|
|
|
rom68k_cmds.line_term = "\r";
|
|
|
|
rom68k_cmds.cmd_end = ".\r";
|
|
|
|
rom68k_cmds.target = &rom68k_ops;
|
|
|
|
rom68k_cmds.stopbits = SERIAL_1_STOPBITS;
|
2002-06-26 17:14:32 +02:00
|
|
|
rom68k_cmds.regnames = NULL;
|
|
|
|
rom68k_cmds.regname = rom68k_regname;
|
1999-07-07 22:19:36 +02:00
|
|
|
rom68k_cmds.magic = MONITOR_OPS_MAGIC;
|
|
|
|
} /* init_rom68k_cmds */
|
1999-04-16 03:35:26 +02:00
|
|
|
|
|
|
|
static void
|
2000-07-30 03:48:28 +02:00
|
|
|
rom68k_open (char *args, int from_tty)
|
1999-04-16 03:35:26 +02:00
|
|
|
{
|
|
|
|
monitor_open (args, &rom68k_cmds, from_tty);
|
|
|
|
}
|
|
|
|
|
2003-06-11 15:16:30 +02:00
|
|
|
extern initialize_file_ftype _initialize_rom68k; /* -Wmissing-prototypes */
|
|
|
|
|
1999-04-16 03:35:26 +02:00
|
|
|
void
|
2000-07-30 03:48:28 +02:00
|
|
|
_initialize_rom68k (void)
|
1999-04-16 03:35:26 +02:00
|
|
|
{
|
1999-07-07 22:19:36 +02:00
|
|
|
init_rom68k_cmds ();
|
1999-04-16 03:35:26 +02:00
|
|
|
init_monitor_ops (&rom68k_ops);
|
|
|
|
|
|
|
|
rom68k_ops.to_shortname = "rom68k";
|
|
|
|
rom68k_ops.to_longname = "Rom68k debug monitor for the IDP Eval board";
|
|
|
|
rom68k_ops.to_doc = "Debug on a Motorola IDP eval board running the ROM68K monitor.\n\
|
|
|
|
Specify the serial device it is connected to (e.g. /dev/ttya).";
|
|
|
|
rom68k_ops.to_open = rom68k_open;
|
|
|
|
|
|
|
|
add_target (&rom68k_ops);
|
|
|
|
}
|