]> git.ipfire.org Git - thirdparty/binutils-gdb.git/commitdiff
This commit was manufactured by cvs2svn to create branch
authornobody <>
Mon, 10 Nov 2003 21:20:45 +0000 (21:20 +0000)
committernobody <>
Mon, 10 Nov 2003 21:20:45 +0000 (21:20 +0000)
'carlton_dictionary-branch'.

Cherrypick from master 2003-11-10 21:20:44 UTC Andrew Cagney <cagney@redhat.com> '2003-11-10  Andrew Cagney  <cagney@redhat.com>':
    gdb/amd64bsd-nat.c
    gdb/amd64nbsd-nat.c
    gdb/amd64nbsd-tdep.c
    gdb/bfd-target.c
    gdb/bfd-target.h
    gdb/config/i386/nbsd64.mh
    gdb/config/i386/nbsd64.mt
    gdb/exec.h
    gdb/glibc-tdep.c
    gdb/glibc-tdep.h
    gdb/i386fbsd-tdep.c
    gdb/regset.h
    gdb/remote-m32r-sdi.c
    gdb/testsuite/gdb.arch/gdb1291.c
    gdb/testsuite/gdb.arch/gdb1291.exp
    gdb/testsuite/gdb.arch/gdb1431.c
    gdb/testsuite/gdb.arch/gdb1431.exp
    gdb/testsuite/gdb.base/gdb1056.exp
    gdb/testsuite/gdb.cp/gdb1355.cc
    gdb/testsuite/gdb.cp/gdb1355.exp
    gdb/testsuite/gdb.threads/switch-threads.c
    gdb/testsuite/gdb.threads/switch-threads.exp
    include/gdb/sim-frv.h

23 files changed:
gdb/amd64bsd-nat.c [new file with mode: 0644]
gdb/amd64nbsd-nat.c [new file with mode: 0644]
gdb/amd64nbsd-tdep.c [new file with mode: 0644]
gdb/bfd-target.c [new file with mode: 0644]
gdb/bfd-target.h [new file with mode: 0644]
gdb/config/i386/nbsd64.mh [new file with mode: 0644]
gdb/config/i386/nbsd64.mt [new file with mode: 0644]
gdb/exec.h [new file with mode: 0644]
gdb/glibc-tdep.c [new file with mode: 0644]
gdb/glibc-tdep.h [new file with mode: 0644]
gdb/i386fbsd-tdep.c [new file with mode: 0644]
gdb/regset.h [new file with mode: 0644]
gdb/remote-m32r-sdi.c [new file with mode: 0644]
gdb/testsuite/gdb.arch/gdb1291.c [new file with mode: 0755]
gdb/testsuite/gdb.arch/gdb1291.exp [new file with mode: 0644]
gdb/testsuite/gdb.arch/gdb1431.c [new file with mode: 0755]
gdb/testsuite/gdb.arch/gdb1431.exp [new file with mode: 0644]
gdb/testsuite/gdb.base/gdb1056.exp [new file with mode: 0644]
gdb/testsuite/gdb.cp/gdb1355.cc [new file with mode: 0644]
gdb/testsuite/gdb.cp/gdb1355.exp [new file with mode: 0644]
gdb/testsuite/gdb.threads/switch-threads.c [new file with mode: 0644]
gdb/testsuite/gdb.threads/switch-threads.exp [new file with mode: 0644]
include/gdb/sim-frv.h [new file with mode: 0644]

diff --git a/gdb/amd64bsd-nat.c b/gdb/amd64bsd-nat.c
new file mode 100644 (file)
index 0000000..6c85f20
--- /dev/null
@@ -0,0 +1,107 @@
+/* Native-dependent code for AMD64 BSD's.
+
+   Copyright 2003 Free Software Foundation, Inc.
+
+   This file is part of GDB.
+
+   This program is free software; you can redistribute it and/or modify
+   it under the terms of the GNU General Public License as published by
+   the Free Software Foundation; either version 2 of the License, or
+   (at your option) any later version.
+
+   This program is distributed in the hope that it will be useful,
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+   GNU General Public License for more details.
+
+   You should have received a copy of the GNU General Public License
+   along with this program; if not, write to the Free Software
+   Foundation, Inc., 59 Temple Place - Suite 330,
+   Boston, MA 02111-1307, USA.  */
+
+#include "defs.h"
+#include "inferior.h"
+#include "regcache.h"
+
+/* We include <signal.h> to make sure `struct fxsave64' is defined on
+   NetBSD, since NetBSD's <machine/reg.h> needs it.  */
+#include "gdb_assert.h"
+#include <signal.h>
+#include <sys/types.h>
+#include <sys/ptrace.h>
+#include <machine/reg.h>
+
+#include "x86-64-tdep.h"
+#include "amd64-nat.h"
+\f
+
+/* Fetch register REGNUM from the inferior.  If REGNUM is -1, do this
+   for all registers (including the floating-point registers).  */
+
+void
+fetch_inferior_registers (int regnum)
+{
+  if (regnum == -1 || amd64_native_gregset_supplies_p (regnum))
+    {
+      struct reg regs;
+
+      if (ptrace (PT_GETREGS, PIDGET (inferior_ptid),
+                 (PTRACE_ARG3_TYPE) &regs, 0) == -1)
+       perror_with_name ("Couldn't get registers");
+
+      amd64_supply_native_gregset (current_regcache, &regs, -1);
+      if (regnum != -1)
+       return;
+    }
+
+  if (regnum == -1 || regnum >= X86_64_ST0_REGNUM)
+    {
+      struct fpreg fpregs;
+
+      if (ptrace (PT_GETFPREGS, PIDGET (inferior_ptid),
+                 (PTRACE_ARG3_TYPE) &fpregs, 0) == -1)
+       perror_with_name ("Couldn't get floating point status");
+
+      x86_64_supply_fxsave (current_regcache, -1, &fpregs);
+    }
+}
+
+/* Store register REGNUM back into the inferior.  If REGNUM is -1, do
+   this for all registers (including the floating-point registers).  */
+
+void
+store_inferior_registers (int regnum)
+{
+  if (regnum == -1 || amd64_native_gregset_supplies_p (regnum))
+    {
+      struct reg regs;
+
+      if (ptrace (PT_GETREGS, PIDGET (inferior_ptid),
+                  (PTRACE_ARG3_TYPE) &regs, 0) == -1)
+        perror_with_name ("Couldn't get registers");
+
+      amd64_collect_native_gregset (current_regcache, &regs, regnum);
+
+      if (ptrace (PT_SETREGS, PIDGET (inferior_ptid),
+                 (PTRACE_ARG3_TYPE) &regs, 0) == -1)
+        perror_with_name ("Couldn't write registers");
+
+      if (regnum != -1)
+       return;
+    }
+
+  if (regnum == -1 || regnum >= X86_64_ST0_REGNUM)
+    {
+      struct fpreg fpregs;
+
+      if (ptrace (PT_GETFPREGS, PIDGET (inferior_ptid),
+                 (PTRACE_ARG3_TYPE) &fpregs, 0) == -1)
+       perror_with_name ("Couldn't get floating point status");
+
+      x86_64_fill_fxsave ((char *) &fpregs, regnum);
+
+      if (ptrace (PT_SETFPREGS, PIDGET (inferior_ptid),
+                 (PTRACE_ARG3_TYPE) &fpregs, 0) == -1)
+       perror_with_name ("Couldn't write floating point status");
+    }
+}
diff --git a/gdb/amd64nbsd-nat.c b/gdb/amd64nbsd-nat.c
new file mode 100644 (file)
index 0000000..4af22fe
--- /dev/null
@@ -0,0 +1,68 @@
+/* Native-dependent code for NetBSD/amd64.
+
+   Copyright 2003 Free Software Foundation, Inc.
+
+   This file is part of GDB.
+
+   This program is free software; you can redistribute it and/or modify
+   it under the terms of the GNU General Public License as published by
+   the Free Software Foundation; either version 2 of the License, or
+   (at your option) any later version.
+
+   This program is distributed in the hope that it will be useful,
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+   GNU General Public License for more details.
+
+   You should have received a copy of the GNU General Public License
+   along with this program; if not, write to the Free Software
+   Foundation, Inc., 59 Temple Place - Suite 330,
+   Boston, MA 02111-1307, USA.  */
+
+#include "defs.h"
+
+#include "gdb_assert.h"
+
+#include "x86-64-tdep.h"
+#include "amd64-nat.h"
+
+/* Mapping between the general-purpose registers in NetBSD/amd64
+   `struct reg' format and GDB's register cache layout for
+   NetBSD/i386.
+
+   Note that most (if not all) NetBSD/amd64 registers are 64-bit,
+   while the NetBSD/i386 registers are all 32-bit, but since we're
+   little-endian we get away with that.  */
+
+/* From <machine/reg.h>.  */
+static int amd64nbsd32_r_reg_offset[] =
+{
+  14 * 8,                      /* %eax */
+  3 * 8,                       /* %ecx */
+  2 * 8,                       /* %edx */
+  13 * 8,                      /* %ebx */
+  24 * 8,                      /* %esp */
+  12 * 8,                      /* %ebp */
+  1 * 8,                       /* %esi */
+  0 * 8,                       /* %edi */
+  21 * 8,                      /* %eip */
+  23 * 8,                      /* %eflags */
+  -1,                          /* %cs */
+  -1,                          /* %ss */
+  18 * 8,                      /* %ds */
+  17 * 8,                      /* %es */
+  16 * 8,                      /* %fs */
+  15 * 8                       /* %gs */
+};
+\f
+
+/* Provide a prototype to silence -Wmissing-prototypes.  */
+void _initialize_amd64nbsd_nat (void);
+
+void
+_initialize_amd64nbsd_nat (void)
+{
+  amd64_native_gregset32_reg_offset = amd64nbsd32_r_reg_offset;
+  amd64_native_gregset32_num_regs = ARRAY_SIZE (amd64nbsd32_r_reg_offset);
+  amd64_native_gregset64_reg_offset = amd64nbsd_r_reg_offset;
+}
diff --git a/gdb/amd64nbsd-tdep.c b/gdb/amd64nbsd-tdep.c
new file mode 100644 (file)
index 0000000..5500ed1
--- /dev/null
@@ -0,0 +1,129 @@
+/* Target-dependent code for NetBSD/amd64.
+
+   Copyright 2003 Free Software Foundation, Inc.
+
+   This file is part of GDB.
+
+   This program is free software; you can redistribute it and/or modify
+   it under the terms of the GNU General Public License as published by
+   the Free Software Foundation; either version 2 of the License, or
+   (at your option) any later version.
+
+   This program is distributed in the hope that it will be useful,
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+   GNU General Public License for more details.
+
+   You should have received a copy of the GNU General Public License
+   along with this program; if not, write to the Free Software
+   Foundation, Inc., 59 Temple Place - Suite 330,
+   Boston, MA 02111-1307, USA.  */
+
+#include "defs.h"
+#include "arch-utils.h"
+#include "frame.h"
+#include "gdbcore.h"
+#include "osabi.h"
+
+#include "gdb_assert.h"
+
+#include "nbsd-tdep.h"
+#include "x86-64-tdep.h"
+
+/* Support for signal handlers.  */
+
+/* Assuming NEXT_FRAME is for a frame following a BSD sigtramp
+   routine, return the address of the associated sigcontext structure.  */
+
+static CORE_ADDR
+amd64nbsd_sigcontext_addr (struct frame_info *next_frame)
+{
+  CORE_ADDR sp;
+
+  /* The stack pointer points at `struct sigcontext' upon entry of a
+     signal trampoline.  */
+  sp = frame_unwind_register_unsigned (next_frame, X86_64_RSP_REGNUM);
+  return sp;
+}
+\f
+/* NetBSD 2.0 or later.  */
+
+/* Mapping between the general-purpose registers in `struct reg'
+   format and GDB's register cache layout.  */
+
+/* From <machine/reg.h>.  */
+int amd64nbsd_r_reg_offset[] =
+{
+  14 * 8,                      /* %rax */
+  13 * 8,                      /* %rbx */
+  3 * 8,                       /* %rcx */
+  2 * 8,                       /* %rdx */
+  1 * 8,                       /* %rsi */
+  0 * 8,                       /* %rdi */
+  12 * 8,                      /* %rbp */
+  24 * 8,                      /* %rsp */
+  4 * 8,                       /* %r8 .. */
+  5 * 8,
+  6 * 8,
+  7 * 8,
+  8 * 8,
+  9 * 8,
+  10 * 8,
+  11 * 8,                      /* ... %r15 */
+  21 * 8,                      /* %rip */
+  23 * 8,                      /* %eflags */
+  18 * 8,                      /* %ds */
+  17 * 8,                      /* %es */
+  16 * 8,                      /* %fs */
+  15 * 8                       /* %gs */
+};
+
+static void
+amd64nbsd_init_abi (struct gdbarch_info info, struct gdbarch *gdbarch)
+{
+  struct gdbarch_tdep *tdep = gdbarch_tdep (gdbarch);
+  int *sc_reg_offset;
+  int i;
+
+  /* Initialize general-purpose register set details first.  */
+  tdep->gregset_reg_offset = amd64nbsd_r_reg_offset;
+  tdep->gregset_num_regs = ARRAY_SIZE (amd64nbsd_r_reg_offset);
+  tdep->sizeof_gregset = 26 * 8;
+
+  x86_64_init_abi (info, gdbarch);
+
+  tdep->jb_pc_offset = 7 * 8;
+
+  /* NetBSD has its own convention for signal trampolines.  */
+  set_gdbarch_pc_in_sigtramp (gdbarch, nbsd_pc_in_sigtramp);
+
+  /* Initialize the array with register offsets in `struct
+     sigcontext'.  This `struct sigcontext' has an sc_mcontext member
+     at offset 32, and in <machine/reg.h> we have an explicit comment
+     saying that `struct reg' is the same as mcontext.__gregs.  */
+  tdep->sc_num_regs = ARRAY_SIZE (amd64nbsd_r_reg_offset);
+  tdep->sc_reg_offset = XCALLOC (tdep->sc_num_regs, int);
+  for (i = 0; i < tdep->sc_num_regs; i++)
+    {
+      if (amd64nbsd_r_reg_offset[i] < 0)
+       tdep->sc_reg_offset[i] = -1;
+      else
+       tdep->sc_reg_offset[i] = 32 + amd64nbsd_r_reg_offset[i];
+    }
+
+  tdep->sigcontext_addr = amd64nbsd_sigcontext_addr;
+}
+\f
+
+/* Provide a prototype to silence -Wmissing-prototypes.  */
+void _initialize_amd64nbsd_tdep (void);
+
+void
+_initialize_amd64nbsd_ndep (void)
+{
+  /* The NetBSD/amd64 native dependent code makes this assumption.  */
+  gdb_assert (ARRAY_SIZE (amd64nbsd_r_reg_offset) == X86_64_NUM_GREGS);
+
+  gdbarch_register_osabi (bfd_arch_i386, bfd_mach_x86_64,
+                         GDB_OSABI_NETBSD_ELF, amd64nbsd_init_abi);
+}
diff --git a/gdb/bfd-target.c b/gdb/bfd-target.c
new file mode 100644 (file)
index 0000000..ee16d85
--- /dev/null
@@ -0,0 +1,131 @@
+/* Very simple "bfd" target, for GDB, the GNU debugger.
+
+   Copyright 2003 Free Software Foundation, Inc.
+
+   This file is part of GDB.
+
+   This program is free software; you can redistribute it and/or modify
+   it under the terms of the GNU General Public License as published by
+   the Free Software Foundation; either version 2 of the License, or
+   (at your option) any later version.
+
+   This program is distributed in the hope that it will be useful,
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+   GNU General Public License for more details.
+
+   You should have received a copy of the GNU General Public License
+   along with this program; if not, write to the Free Software
+   Foundation, Inc., 59 Temple Place - Suite 330,
+   Boston, MA 02111-1307, USA.  */
+
+#include "defs.h"
+#include "target.h"
+#include "bfd-target.h"
+#include "gdb_assert.h"
+#include "gdb_string.h"
+
+/* Locate all mappable sections of a BFD file, filling in a target
+   section for each.  */
+
+struct section_closure
+{
+  struct section_table *end;
+};
+
+static void
+add_to_section_table (struct bfd *abfd, struct bfd_section *asect,
+                     void *closure)
+{
+  struct section_closure *pp = closure;
+  flagword aflag;
+
+  /* NOTE: cagney/2003-10-22: Is this pruning useful?  */
+  aflag = bfd_get_section_flags (abfd, asect);
+  if (!(aflag & SEC_ALLOC))
+    return;
+  if (bfd_section_size (abfd, asect) == 0)
+    return;
+  pp->end->bfd = abfd;
+  pp->end->the_bfd_section = asect;
+  pp->end->addr = bfd_section_vma (abfd, asect);
+  pp->end->endaddr = pp->end->addr + bfd_section_size (abfd, asect);
+  pp->end++;
+}
+
+void
+build_target_sections_from_bfd (struct target_ops *targ, struct bfd *abfd)
+{
+  unsigned count;
+  struct section_table *start;
+  struct section_closure cl;
+
+  count = bfd_count_sections (abfd);
+  target_resize_to_sections (targ, count);
+  start = targ->to_sections;
+  cl.end = targ->to_sections;
+  bfd_map_over_sections (abfd, add_to_section_table, &cl);
+  gdb_assert (cl.end - start <= count);
+}
+
+LONGEST
+target_bfd_xfer_partial (struct target_ops *ops,
+                        enum target_object object,
+                        const char *annex, void *readbuf,
+                        const void *writebuf, ULONGEST offset, LONGEST len)
+{
+  switch (object)
+    {
+    case TARGET_OBJECT_MEMORY:
+      {
+       struct section_table *s = target_section_by_addr (ops, offset);
+       if (s == NULL)
+         return -1;
+       /* If the length extends beyond the section, truncate it.  Be
+           careful to not suffer from overflow (wish S contained a
+           length).  */
+       if ((offset - s->addr + len) > (s->endaddr - s->addr))
+         len = (s->endaddr - s->addr) - (offset - s->addr);
+       if (readbuf != NULL
+           && !bfd_get_section_contents (s->bfd, s->the_bfd_section,
+                                         readbuf, offset - s->addr, len))
+         return -1;
+#if 1
+       if (writebuf != NULL)
+         return -1;
+#else
+       /* FIXME: cagney/2003-10-31: The BFD interface doesn't yet
+           take a const buffer.  */
+       if (writebuf != NULL
+           && !bfd_set_section_contents (s->bfd, s->the_bfd_section,
+                                         writebuf, offset - s->addr, len))
+         return -1;
+#endif
+       return len;
+      }
+    default:
+      return -1;
+    }
+}
+
+void
+target_bfd_xclose (struct target_ops *t, int quitting)
+{
+  bfd_close (t->to_data);
+  xfree (t->to_sections);
+  xfree (t);
+}
+
+struct target_ops *
+target_bfd_reopen (struct bfd *bfd)
+{
+  struct target_ops *t = XZALLOC (struct target_ops);
+  t->to_shortname = "bfd";
+  t->to_longname = "BFD backed target";
+  t->to_doc = "You should never see this";
+  t->to_xfer_partial = target_bfd_xfer_partial;
+  t->to_xclose = target_bfd_xclose;
+  t->to_data = bfd;
+  build_target_sections_from_bfd (t, bfd);
+  return t;
+}
diff --git a/gdb/bfd-target.h b/gdb/bfd-target.h
new file mode 100644 (file)
index 0000000..61a51c8
--- /dev/null
@@ -0,0 +1,39 @@
+/* Very simple "bfd" target, for GDB, the GNU debugger.
+
+   Copyright 2003 Free Software Foundation, Inc.
+
+   This file is part of GDB.
+
+   This program is free software; you can redistribute it and/or modify
+   it under the terms of the GNU General Public License as published by
+   the Free Software Foundation; either version 2 of the License, or
+   (at your option) any later version.
+
+   This program is distributed in the hope that it will be useful,
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+   GNU General Public License for more details.
+
+   You should have received a copy of the GNU General Public License
+   along with this program; if not, write to the Free Software
+   Foundation, Inc., 59 Temple Place - Suite 330,
+   Boston, MA 02111-1307, USA.  */
+
+#ifndef BFD_TARGET_H
+#define BFD_TARGET_H
+
+struct bfd;
+struct target_ops;
+
+/* Given an existing BFD, re-open it as a "struct target_ops".  On
+   close, it will also close the corresponding BFD (which is like
+   freopen and fdopen).  */
+struct target_ops *target_bfd_reopen (struct bfd *bfd);
+
+/* Map over ABFD's sections, creating corresponding entries in the
+   target's section table.  */
+
+void build_target_sections_from_bfd (struct target_ops *targ,
+                                    struct bfd *abfd);
+
+#endif
diff --git a/gdb/config/i386/nbsd64.mh b/gdb/config/i386/nbsd64.mh
new file mode 100644 (file)
index 0000000..5acd167
--- /dev/null
@@ -0,0 +1,7 @@
+# Host: NetBSD/amd64
+
+XM_FILE= xm-i386.h
+
+NAT_FILE= nm-nbsd.h
+# NOTE: Do not spread NATDEPFILES over several lines - it hurts BSD make.
+NATDEPFILES= fork-child.o infptrace.o inftarg.o solib.o solib-svr4.o solib-legacy.o corelow.o amd64-nat.o amd64bsd-nat.o amd64nbsd-nat.o
diff --git a/gdb/config/i386/nbsd64.mt b/gdb/config/i386/nbsd64.mt
new file mode 100644 (file)
index 0000000..6d73660
--- /dev/null
@@ -0,0 +1,2 @@
+# Target: NetBSD/amd64
+TDEPFILES= x86-64-tdep.o amd64nbsd-tdep.o i386-tdep.o i387-tdep.o nbsd-tdep.o
diff --git a/gdb/exec.h b/gdb/exec.h
new file mode 100644 (file)
index 0000000..e9c2d17
--- /dev/null
@@ -0,0 +1,39 @@
+/* Work with executable files, for GDB, the GNU debugger.
+
+   Copyright 2003 Free Software Foundation, Inc.
+
+   This file is part of GDB.
+
+   This program is free software; you can redistribute it and/or modify
+   it under the terms of the GNU General Public License as published by
+   the Free Software Foundation; either version 2 of the License, or
+   (at your option) any later version.
+
+   This program is distributed in the hope that it will be useful,
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+   GNU General Public License for more details.
+
+   You should have received a copy of the GNU General Public License
+   along with this program; if not, write to the Free Software
+   Foundation, Inc., 59 Temple Place - Suite 330,
+   Boston, MA 02111-1307, USA.  */
+
+#ifndef EXEC_H
+#define EXEC_H
+
+#include "target.h"
+
+struct section_table;
+struct target_ops;
+struct bfd;
+
+struct target_ops exec_ops;
+
+/* Builds a section table, given args BFD, SECTABLE_PTR, SECEND_PTR.
+   Returns 0 if OK, 1 on error.  */
+
+extern int build_section_table (struct bfd *, struct section_table **,
+                               struct section_table **);
+
+#endif
diff --git a/gdb/glibc-tdep.c b/gdb/glibc-tdep.c
new file mode 100644 (file)
index 0000000..46aa749
--- /dev/null
@@ -0,0 +1,101 @@
+/* Target-dependent code for the GNU C Library (glibc).
+
+   Copyright 2002, 2003 Free Software Foundation, Inc.
+
+   This file is part of GDB.
+
+   This program is free software; you can redistribute it and/or modify
+   it under the terms of the GNU General Public License as published by
+   the Free Software Foundation; either version 2 of the License, or
+   (at your option) any later version.
+
+   This program is distributed in the hope that it will be useful,
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+   GNU General Public License for more details.
+
+   You should have received a copy of the GNU General Public License
+   along with this program; if not, write to the Free Software
+   Foundation, Inc., 59 Temple Place - Suite 330,
+   Boston, MA 02111-1307, USA.  */
+
+#include "defs.h"
+#include "frame.h"
+#include "symtab.h"
+#include "symfile.h"
+#include "objfiles.h"
+
+#include "glibc-tdep.h"
+
+/* Calling functions in shared libraries.  */
+
+/* Find the minimal symbol named NAME, and return both the minsym
+   struct and its objfile.  This probably ought to be in minsym.c, but
+   everything there is trying to deal with things like C++ and
+   SOFUN_ADDRESS_MAYBE_TURQUOISE, ...  Since this is so simple, it may
+   be considered too special-purpose for general consumption.  */
+
+static struct minimal_symbol *
+find_minsym_and_objfile (char *name, struct objfile **objfile_p)
+{
+  struct objfile *objfile;
+
+  ALL_OBJFILES (objfile)
+    {
+      struct minimal_symbol *msym;
+
+      ALL_OBJFILE_MSYMBOLS (objfile, msym)
+       {
+         if (SYMBOL_LINKAGE_NAME (msym)
+             && strcmp (SYMBOL_LINKAGE_NAME (msym), name) == 0)
+           {
+             *objfile_p = objfile;
+             return msym;
+           }
+       }
+    }
+
+  return 0;
+}
+
+/* See the comments for SKIP_SOLIB_RESOLVER at the top of infrun.c.
+   This function:
+   1) decides whether a PLT has sent us into the linker to resolve
+      a function reference, and 
+   2) if so, tells us where to set a temporary breakpoint that will
+      trigger when the dynamic linker is done.  */
+
+CORE_ADDR
+glibc_skip_solib_resolver (CORE_ADDR pc)
+{
+  /* The GNU dynamic linker is part of the GNU C library, and is used
+     by all GNU systems (GNU/Hurd, GNU/Linux).  An unresolved PLT
+     entry points to "_dl_runtime_resolve", which calls "fixup" to
+     patch the PLT, and then passes control to the function.
+
+     We look for the symbol `_dl_runtime_resolve', and find `fixup' in
+     the same objfile.  If we are at the entry point of `fixup', then
+     we set a breakpoint at the return address (at the top of the
+     stack), and continue.
+  
+     It's kind of gross to do all these checks every time we're
+     called, since they don't change once the executable has gotten
+     started.  But this is only a temporary hack --- upcoming versions
+     of GNU/Linux will provide a portable, efficient interface for
+     debugging programs that use shared libraries.  */
+
+  struct objfile *objfile;
+  struct minimal_symbol *resolver 
+    = find_minsym_and_objfile ("_dl_runtime_resolve", &objfile);
+
+  if (resolver)
+    {
+      struct minimal_symbol *fixup
+       = lookup_minimal_symbol ("fixup", NULL, objfile);
+
+      if (fixup && SYMBOL_VALUE_ADDRESS (fixup) == pc)
+       return frame_pc_unwind (get_current_frame ());
+    }
+
+  return 0;
+}      
diff --git a/gdb/glibc-tdep.h b/gdb/glibc-tdep.h
new file mode 100644 (file)
index 0000000..31c77b9
--- /dev/null
@@ -0,0 +1,27 @@
+/* Target-dependent code for the GNU C Library (glibc).
+
+   Copyright 2002, 2003 Free Software Foundation, Inc.
+
+   This file is part of GDB.
+
+   This program is free software; you can redistribute it and/or modify
+   it under the terms of the GNU General Public License as published by
+   the Free Software Foundation; either version 2 of the License, or
+   (at your option) any later version.
+
+   This program is distributed in the hope that it will be useful,
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+   GNU General Public License for more details.
+
+   You should have received a copy of the GNU General Public License
+   along with this program; if not, write to the Free Software
+   Foundation, Inc., 59 Temple Place - Suite 330,
+   Boston, MA 02111-1307, USA.  */
+
+#ifndef GLIBC_TDEP_H
+#define GLIBC_TDEP_H
+
+extern CORE_ADDR glibc_skip_solib_resolver (CORE_ADDR);
+
+#endif /* glibc-tdep.h */
diff --git a/gdb/i386fbsd-tdep.c b/gdb/i386fbsd-tdep.c
new file mode 100644 (file)
index 0000000..786de7d
--- /dev/null
@@ -0,0 +1,171 @@
+/* Target-dependent code for FreeBSD/i386.
+
+   Copyright 2003 Free Software Foundation, Inc.
+
+   This file is part of GDB.
+
+   This program is free software; you can redistribute it and/or modify
+   it under the terms of the GNU General Public License as published by
+   the Free Software Foundation; either version 2 of the License, or
+   (at your option) any later version.
+
+   This program is distributed in the hope that it will be useful,
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+   GNU General Public License for more details.
+
+   You should have received a copy of the GNU General Public License
+   along with this program; if not, write to the Free Software
+   Foundation, Inc., 59 Temple Place - Suite 330,
+   Boston, MA 02111-1307, USA.  */
+
+#include "defs.h"
+#include "arch-utils.h"
+#include "osabi.h"
+
+#include "i386-tdep.h"
+#include "i387-tdep.h"
+
+/* FreeBSD 3.0-RELEASE or later.  */
+
+/* From <machine/reg.h>.  */
+static int i386fbsd_r_reg_offset[] =
+{
+  9 * 4, 8 * 4, 7 * 4, 6 * 4,  /* %eax, %ecx, %edx, %ebx */
+  15 * 4, 4 * 4,               /* %esp, %ebp */
+  3 * 4, 2 * 4,                        /* %esi, %edi */
+  12 * 4, 14 * 4,              /* %eip, %eflags */
+  13 * 4, 16 * 4,              /* %cs, %ss */
+  1 * 4, 0 * 4, -1, -1         /* %ds, %es, %fs, %gs */
+};
+
+CORE_ADDR i386fbsd_sigtramp_start = 0xbfbfdf20;
+CORE_ADDR i386fbsd_sigtramp_end = 0xbfbfdff0;
+
+/* From <machine/signal.h>.  */
+static int i386fbsd_sc_reg_offset[] =
+{
+  8 + 14 * 4,                  /* %eax */
+  8 + 13 * 4,                  /* %ecx */
+  8 + 12 * 4,                  /* %edx */
+  8 + 11 * 4,                  /* %ebx */
+  8 + 0 * 4,                    /* %esp */
+  8 + 1 * 4,                    /* %ebp */
+  8 + 10 * 4,                   /* %esi */
+  8 + 9 * 4,                    /* %edi */
+  8 + 3 * 4,                    /* %eip */
+  8 + 4 * 4,                    /* %eflags */
+  8 + 7 * 4,                    /* %cs */
+  8 + 8 * 4,                    /* %ss */
+  8 + 6 * 4,                    /* %ds */
+  8 + 5 * 4,                    /* %es */
+  8 + 15 * 4,                  /* %fs */
+  8 + 16 * 4                   /* %gs */
+};
+
+static void
+i386fbsdaout_init_abi (struct gdbarch_info info, struct gdbarch *gdbarch)
+{
+  struct gdbarch_tdep *tdep = gdbarch_tdep (gdbarch);
+
+  /* Obviously FreeBSD is BSD-based.  */
+  i386bsd_init_abi (info, gdbarch);
+
+  /* FreeBSD has a different `struct reg', and reserves some space for
+     its FPU emulator in `struct fpreg'.  */
+  tdep->gregset_reg_offset = i386fbsd_r_reg_offset;
+  tdep->gregset_num_regs = ARRAY_SIZE (i386fbsd_r_reg_offset);
+  tdep->sizeof_gregset = 18 * 4;
+  tdep->sizeof_fpregset = 176;
+
+  /* FreeBSD uses -freg-struct-return by default.  */
+  tdep->struct_return = reg_struct_return;
+
+  /* FreeBSD uses a different memory layout.  */
+  tdep->sigtramp_start = i386fbsd_sigtramp_start;
+  tdep->sigtramp_end = i386fbsd_sigtramp_end;
+
+  /* FreeBSD has a more complete `struct sigcontext'.  */
+  tdep->sc_reg_offset = i386fbsd_sc_reg_offset;
+  tdep->sc_num_regs = ARRAY_SIZE (i386fbsd_sc_reg_offset);
+}
+
+static void
+i386fbsd_init_abi (struct gdbarch_info info, struct gdbarch *gdbarch)
+{
+  /* It's almost identical to FreeBSD a.out.  */
+  i386fbsdaout_init_abi (info, gdbarch);
+
+  /* Except that it uses ELF.  */
+  i386_elf_init_abi (info, gdbarch);
+
+  /* FreeBSD ELF uses SVR4-style shared libraries.  */
+  set_gdbarch_in_solib_call_trampoline (gdbarch,
+                                       generic_in_solib_call_trampoline);
+}
+
+/* FreeBSD 4.0-RELEASE or later.  */
+
+/* From <machine/reg.h>.  */
+static int i386fbsd4_r_reg_offset[] =
+{
+  10 * 4, 9 * 4, 8 * 4, 7 * 4, /* %eax, %ecx, %edx, %ebx */
+  16 * 4, 5 * 4,               /* %esp, %ebp */
+  4 * 4, 3 * 4,                        /* %esi, %edi */
+  13 * 4, 15 * 4,              /* %eip, %eflags */
+  14 * 4, 17 * 4,              /* %cs, %ss */
+  2 * 4, 1 * 4, 0 * 4, 18 * 4  /* %ds, %es, %fs, %gs */
+};
+
+/* From <machine/signal.h>.  */
+int i386fbsd4_sc_reg_offset[] =
+{
+  20 + 11 * 4,                 /* %eax */
+  20 + 10 * 4,                 /* %ecx */
+  20 + 9 * 4,                  /* %edx */
+  20 + 8 * 4,                  /* %ebx */
+  20 + 17 * 4,                 /* %esp */
+  20 + 6 * 4,                  /* %ebp */
+  20 + 5 * 4,                  /* %esi */
+  20 + 4 * 4,                  /* %edi */
+  20 + 14 * 4,                 /* %eip */
+  20 + 16 * 4,                 /* %eflags */
+  20 + 15 * 4,                 /* %cs */
+  20 + 18 * 4,                 /* %ss */
+  20 + 3 * 4,                  /* %ds */
+  20 + 2 * 4,                  /* %es */
+  20 + 1 * 4,                  /* %fs */
+  20 + 0 * 4                   /* %gs */
+};
+
+static void
+i386fbsd4_init_abi (struct gdbarch_info info, struct gdbarch *gdbarch)
+{
+  struct gdbarch_tdep *tdep = gdbarch_tdep (gdbarch);
+
+  /* Inherit stuff from older releases.  We assume that FreeBSD
+     4.0-RELEASE always uses ELF.  */
+  i386fbsd_init_abi (info, gdbarch);
+
+  /* FreeBSD 4.0 introduced a new `struct reg'.  */
+  tdep->gregset_reg_offset = i386fbsd4_r_reg_offset;
+  tdep->gregset_num_regs = ARRAY_SIZE (i386fbsd4_r_reg_offset);
+  tdep->sizeof_gregset = 19 * 4;
+
+  /* FreeBSD 4.0 introduced a new `struct sigcontext'.  */
+  tdep->sc_reg_offset = i386fbsd4_sc_reg_offset;
+  tdep->sc_num_regs = ARRAY_SIZE (i386fbsd4_sc_reg_offset);
+}
+
+\f
+/* Provide a prototype to silence -Wmissing-prototypes.  */
+void _initialize_i386fbsd_tdep (void);
+
+void
+_initialize_i386fbsd_tdep (void)
+{
+  gdbarch_register_osabi (bfd_arch_i386, 0, GDB_OSABI_FREEBSD_AOUT,
+                         i386fbsdaout_init_abi);
+  gdbarch_register_osabi (bfd_arch_i386, 0, GDB_OSABI_FREEBSD_ELF,
+                         i386fbsd4_init_abi);
+}
diff --git a/gdb/regset.h b/gdb/regset.h
new file mode 100644 (file)
index 0000000..6172f0f
--- /dev/null
@@ -0,0 +1,41 @@
+/* Manage register sets.
+
+   Copyright 2003 Free Software Foundation, Inc.
+
+   This file is part of GDB.
+
+   This program is free software; you can redistribute it and/or modify
+   it under the terms of the GNU General Public License as published by
+   the Free Software Foundation; either version 2 of the License, or
+   (at your option) any later version.
+
+   This program is distributed in the hope that it will be useful,
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+   GNU General Public License for more details.
+
+   You should have received a copy of the GNU General Public License
+   along with this program; if not, write to the Free Software
+   Foundation, Inc., 59 Temple Place - Suite 330,
+   Boston, MA 02111-1307, USA.  */
+
+#ifndef REGSET_H
+#define REGSET_H 1
+
+struct gdbarch;
+struct regcache;
+
+/* Data structure describing a register set.  */
+
+struct regset
+{
+  /* Data pointer for private use by the methods below, presumably
+     providing some sort of description of the register set.  */
+  const void *descr;
+
+  /* Function supplying a register set to a register cache.  */
+  void (*supply_regset) (const struct regset *, struct regcache *,
+                        int, const void *, size_t);
+};
+
+#endif /* regset.h */
diff --git a/gdb/remote-m32r-sdi.c b/gdb/remote-m32r-sdi.c
new file mode 100644 (file)
index 0000000..7f0b90c
--- /dev/null
@@ -0,0 +1,1673 @@
+/* Remote debugging interface for M32R/SDI.
+
+   Copyright 2003 Free Software Foundation, Inc.
+
+   Contributed by Renesas Technology Co.
+   Written by Kei Sakamoto <sakamoto.kei@renesas.com>.
+
+   This file is part of GDB.
+
+   This program is free software; you can redistribute it and/or modify
+   it under the terms of the GNU General Public License as published by
+   the Free Software Foundation; either version 2 of the License, or
+   (at your option) any later version.
+
+   This program is distributed in the hope that it will be useful,
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+   GNU General Public License for more details.
+
+   You should have received a copy of the GNU General Public License
+   along with this program; if not, write to the Free Software
+   Foundation, Inc., 59 Temple Place - Suite 330,
+   Boston, MA 02111-1307, USA.  */
+
+#include "defs.h"
+#include "gdbcmd.h"
+#include "gdbcore.h"
+#include "inferior.h"
+#include "target.h"
+#include "regcache.h"
+#include "gdb_string.h"
+#include <ctype.h>
+#include <signal.h>
+#include <netinet/in.h>
+#include <sys/types.h>
+#include <sys/time.h>
+#include <signal.h>
+#include <time.h>
+
+
+#include "serial.h"
+
+/* Descriptor for I/O to remote machine.  */
+
+static struct serial *sdi_desc = NULL;
+
+#define SDI_TIMEOUT 30
+
+
+#define SDIPORT 3232
+
+static char chip_name[64];
+
+static int step_mode;
+static unsigned long last_pc_addr = 0xffffffff;
+static unsigned char last_pc_addr_data[2];
+
+static int mmu_on = 0;
+
+static int use_ib_breakpoints = 1;
+
+#define MAX_BREAKPOINTS 1024
+static int max_ib_breakpoints;
+static unsigned long bp_address[MAX_BREAKPOINTS];
+static unsigned char bp_data[MAX_BREAKPOINTS][4];
+static const unsigned char ib_bp_entry_enable[] = {
+  0x00, 0x00, 0x00, 0x06
+};
+static const unsigned char ib_bp_entry_disable[] = {
+  0x00, 0x00, 0x00, 0x00
+};
+
+/* dbt -> nop */
+static const unsigned char dbt_bp_entry[] = {
+  0x10, 0xe0, 0x70, 0x00
+};
+
+#define MAX_ACCESS_BREAKS 4
+static int max_access_breaks;
+static unsigned long ab_address[MAX_ACCESS_BREAKS];
+static unsigned int ab_type[MAX_ACCESS_BREAKS];
+static unsigned int ab_size[MAX_ACCESS_BREAKS];
+static CORE_ADDR hit_watchpoint_addr = 0;
+
+static int interrupted = 0;
+
+/* Forward data declarations */
+extern struct target_ops m32r_ops;
+
+
+/* Commands */
+#define SDI_OPEN                 1
+#define SDI_CLOSE                2
+#define SDI_RELEASE              3
+#define SDI_READ_CPU_REG         4
+#define SDI_WRITE_CPU_REG        5
+#define SDI_READ_MEMORY          6
+#define SDI_WRITE_MEMORY         7
+#define SDI_EXEC_CPU             8
+#define SDI_STOP_CPU             9
+#define SDI_WAIT_FOR_READY      10
+#define SDI_GET_ATTR            11
+#define SDI_SET_ATTR            12
+#define SDI_STATUS              13
+
+/* Attributes */
+#define SDI_ATTR_NAME            1
+#define SDI_ATTR_BRK             2
+#define SDI_ATTR_ABRK            3
+#define SDI_ATTR_CACHE           4
+#define SDI_CACHE_TYPE_M32102    0
+#define SDI_CACHE_TYPE_CHAOS     1
+#define SDI_ATTR_MEM_ACCESS      5
+#define SDI_MEM_ACCESS_DEBUG_DMA 0
+#define SDI_MEM_ACCESS_MON_CODE  1
+
+/* Registers */
+#define SDI_REG_R0               0
+#define SDI_REG_R1               1
+#define SDI_REG_R2               2
+#define SDI_REG_R3               3
+#define SDI_REG_R4               4
+#define SDI_REG_R5               5
+#define SDI_REG_R6               6
+#define SDI_REG_R7               7
+#define SDI_REG_R8               8
+#define SDI_REG_R9               9
+#define SDI_REG_R10             10
+#define SDI_REG_R11             11
+#define SDI_REG_R12             12
+#define SDI_REG_FP              13
+#define SDI_REG_LR              14
+#define SDI_REG_SP              15
+#define SDI_REG_PSW             16
+#define SDI_REG_CBR             17
+#define SDI_REG_SPI             18
+#define SDI_REG_SPU             19
+#define SDI_REG_CR4             20
+#define SDI_REG_EVB             21
+#define SDI_REG_BPC             22
+#define SDI_REG_CR7             23
+#define SDI_REG_BBPSW           24
+#define SDI_REG_CR9             25
+#define SDI_REG_CR10            26
+#define SDI_REG_CR11            27
+#define SDI_REG_CR12            28
+#define SDI_REG_WR              29
+#define SDI_REG_BBPC            30
+#define SDI_REG_PBP             31
+#define SDI_REG_ACCH            32
+#define SDI_REG_ACCL            33
+#define SDI_REG_ACC1H           34
+#define SDI_REG_ACC1L           35
+
+
+/* Low level communication functions */
+
+/* Check an ack packet from the target */
+static int
+get_ack (void)
+{
+  int c;
+
+  if (!sdi_desc) 
+    return -1;
+
+  c = serial_readchar (sdi_desc, SDI_TIMEOUT);
+
+  if (c < 0)
+    return -1;
+
+  if (c != '+')                /* error */
+    return -1;
+
+  return 0;
+}
+
+/* Send data to the target and check an ack packet */
+static int
+send_data (void *buf, int len)
+{
+  int ret;
+
+  if (!sdi_desc) 
+    return -1;
+
+  if (serial_write (sdi_desc, buf, len) != 0)
+    return -1;
+
+  if (get_ack () == -1)
+    return -1;
+
+  return len;
+}
+
+/* Receive data from the target */
+static int
+recv_data (void *buf, int len)
+{
+  int total = 0;
+  int c;
+
+  if (!sdi_desc) 
+    return -1;
+
+  while (total < len)
+    {
+      c = serial_readchar (sdi_desc, SDI_TIMEOUT);
+
+      if (c < 0)
+       return -1;
+
+      ((unsigned char *) buf)[total++] = c;
+    }
+
+  return len;
+}
+
+/* Store unsigned long parameter on packet */
+static void
+store_long_parameter (void *buf, long val)
+{
+  val = htonl (val);
+  memcpy (buf, &val, 4);
+}
+
+/* Check if MMU is on */
+static void
+check_mmu_status (void)
+{
+  unsigned long val;
+  unsigned char buf[2];
+
+  /* Read PC address */
+  buf[0] = SDI_READ_CPU_REG;
+  buf[1] = SDI_REG_BPC;
+  if (send_data (buf, 2) == -1)
+    return;
+  recv_data (&val, 4);
+  val = ntohl (val);
+  if ((val & 0xc0000000) == 0x80000000)
+    {
+      mmu_on = 1;
+      return;
+    }
+
+  /* Read EVB address */
+  buf[0] = SDI_READ_CPU_REG;
+  buf[1] = SDI_REG_EVB;
+  if (send_data (buf, 2) == -1)
+    return;
+  recv_data (&val, 4);
+  val = ntohl (val);
+  if ((val & 0xc0000000) == 0x80000000)
+    {
+      mmu_on = 1;
+      return;
+    }
+
+  mmu_on = 0;
+}
+
+
+/* This is called not only when we first attach, but also when the
+   user types "run" after having attached.  */
+static void
+m32r_create_inferior (char *execfile, char *args, char **env)
+{
+  CORE_ADDR entry_pt;
+
+  if (args && *args)
+    error ("Cannot pass arguments to remote STDEBUG process");
+
+  if (execfile == 0 || exec_bfd == 0)
+    error ("No executable file specified");
+
+  if (remote_debug)
+    fprintf_unfiltered (gdb_stdlog, "m32r_create_inferior(%s,%s)\n", execfile,
+                       args);
+
+  entry_pt = bfd_get_start_address (exec_bfd);
+
+  /* The "process" (board) is already stopped awaiting our commands, and
+     the program is already downloaded.  We just set its PC and go.  */
+
+  clear_proceed_status ();
+
+  /* Tell wait_for_inferior that we've started a new process.  */
+  init_wait_for_inferior ();
+
+  /* Set up the "saved terminal modes" of the inferior
+     based on what modes we are starting it with.  */
+  target_terminal_init ();
+
+  /* Install inferior's terminal modes.  */
+  target_terminal_inferior ();
+
+  proceed (entry_pt, TARGET_SIGNAL_DEFAULT, 0);
+}
+
+/* Open a connection to a remote debugger.
+   NAME is the filename used for communication.  */
+
+static void
+m32r_open (char *args, int from_tty)
+{
+  struct hostent *host_ent;
+  struct sockaddr_in server_addr;
+  char *port_str, hostname[256];
+  int port;
+  unsigned char buf[2];
+  int i, n;
+  int yes = 1;
+
+  if (remote_debug)
+    fprintf_unfiltered (gdb_stdlog, "m32r_open(%d)\n", from_tty);
+
+  target_preopen (from_tty);
+
+  push_target (&m32r_ops);
+
+  if (args == NULL)
+    sprintf (hostname, "localhost:%d", SDIPORT);
+  else
+    {
+      port_str = strchr (args, ':');
+      if (port_str == NULL)
+        sprintf (hostname, "%s:%d", args, SDIPORT);
+      else
+       strcpy (hostname, args);
+    }
+
+  sdi_desc = serial_open (hostname);
+  if (!sdi_desc)
+    error ("Connection refused\n");
+
+  if (get_ack () == -1)
+    error ("Cannot connect to SDI target\n");
+
+  buf[0] = SDI_OPEN;
+  if (send_data (buf, 1) == -1)
+    error ("Cannot connect to SDI target\n");
+
+  /* Get maximum number of ib breakpoints */
+  buf[0] = SDI_GET_ATTR;
+  buf[1] = SDI_ATTR_BRK;
+  send_data (buf, 2);
+  recv_data (buf, 1);
+  max_ib_breakpoints = buf[0];
+  if (remote_debug)
+    printf_filtered ("Max IB Breakpoints = %d\n", max_ib_breakpoints);
+
+  /* Initialize breakpoints. */
+  for (i = 0; i < MAX_BREAKPOINTS; i++)
+    bp_address[i] = 0xffffffff;
+
+  /* Get maximum number of access breaks. */
+  buf[0] = SDI_GET_ATTR;
+  buf[1] = SDI_ATTR_ABRK;
+  send_data (buf, 2);
+  recv_data (buf, 1);
+  max_access_breaks = buf[0];
+  if (remote_debug)
+    printf_filtered ("Max Access Breaks = %d\n", max_access_breaks);
+
+  /* Initialize access breask. */
+  for (i = 0; i < MAX_ACCESS_BREAKS; i++)
+    ab_address[i] = 0x00000000;
+
+  check_mmu_status ();
+
+  /* Get the name of chip on target board. */
+  buf[0] = SDI_GET_ATTR;
+  buf[1] = SDI_ATTR_NAME;
+  send_data (buf, 2);
+  recv_data (chip_name, 64);
+
+  if (from_tty)
+    printf_filtered ("Remote %s connected to %s\n", target_shortname,
+                    chip_name);
+}
+
+/* Close out all files and local state before this target loses control. */
+
+static void
+m32r_close (int quitting)
+{
+  unsigned char buf[1];
+
+  if (remote_debug)
+    fprintf_unfiltered (gdb_stdlog, "m32r_close(%d)\n", quitting);
+
+  if (sdi_desc)
+    {
+      buf[0] = SDI_CLOSE;
+      send_data (buf, 1);
+      serial_close (sdi_desc);
+      sdi_desc = NULL;
+    }
+
+  inferior_ptid = null_ptid;
+  return;
+}
+
+/* Tell the remote machine to resume.  */
+
+static void
+m32r_resume (ptid_t ptid, int step, enum target_signal sig)
+{
+  unsigned long pc_addr, bp_addr, ab_addr;
+  unsigned char buf[13];
+  int i;
+
+  if (remote_debug)
+    {
+      if (step)
+       fprintf_unfiltered (gdb_stdlog, "\nm32r_resume(step)\n");
+      else
+       fprintf_unfiltered (gdb_stdlog, "\nm32r_resume(cont)\n");
+    }
+
+  check_mmu_status ();
+
+  pc_addr = read_pc ();
+  if (remote_debug)
+    fprintf_unfiltered (gdb_stdlog, "pc <= 0x%lx\n", pc_addr);
+
+  /* At pc address there is a parallel instruction with +2 offset,
+     so we have to make it a serial instruction or avoid it. */
+  if (pc_addr == last_pc_addr)
+    {
+      /* Avoid a parallel nop. */
+      if (last_pc_addr_data[0] == 0xf0 && last_pc_addr_data[1] == 0x00)
+       {
+         pc_addr += 2;
+         /* Now we can forget this instruction. */
+         last_pc_addr = 0xffffffff;
+       }
+      /* Clear a parallel bit. */
+      else
+       {
+         buf[0] = SDI_WRITE_MEMORY;
+         if (TARGET_BYTE_ORDER == BFD_ENDIAN_BIG)
+           store_long_parameter (buf + 1, pc_addr);
+         else
+           store_long_parameter (buf + 1, pc_addr - 1);
+         store_long_parameter (buf + 5, 1);
+         buf[9] = last_pc_addr_data[0] & 0x7f;
+         send_data (buf, 10);
+       }
+    }
+
+  /* Set PC. */
+  buf[0] = SDI_WRITE_CPU_REG;
+  buf[1] = SDI_REG_BPC;
+  store_long_parameter (buf + 2, pc_addr);
+  send_data (buf, 6);
+
+  /* step mode. */
+  step_mode = step;
+  if (step)
+    {
+      /* Set PBP. */
+      buf[0] = SDI_WRITE_CPU_REG;
+      buf[1] = SDI_REG_PBP;
+      store_long_parameter (buf + 2, pc_addr | 1);
+      send_data (buf, 6);
+    }
+  else
+    {
+      int ib_breakpoints;
+
+      if (use_ib_breakpoints)
+       ib_breakpoints = max_ib_breakpoints;
+      else
+       ib_breakpoints = 0;
+
+      /* Set ib breakpoints. */
+      for (i = 0; i < ib_breakpoints; i++)
+       {
+         bp_addr = bp_address[i];
+         if (bp_addr != 0xffffffff && bp_addr != pc_addr)
+           {
+             /* Set PBP. */
+             buf[0] = SDI_WRITE_MEMORY;
+             store_long_parameter (buf + 1, 0xffff8000 + 4 * i);
+             store_long_parameter (buf + 5, 4);
+             if (TARGET_BYTE_ORDER == BFD_ENDIAN_BIG)
+               {
+                 buf[9] = ib_bp_entry_enable[0];
+                 buf[10] = ib_bp_entry_enable[1];
+                 buf[11] = ib_bp_entry_enable[2];
+                 buf[12] = ib_bp_entry_enable[3];
+               }
+             else
+               {
+                 buf[9] = ib_bp_entry_enable[3];
+                 buf[10] = ib_bp_entry_enable[2];
+                 buf[11] = ib_bp_entry_enable[1];
+                 buf[12] = ib_bp_entry_enable[0];
+               }
+             send_data (buf, 13);
+
+             buf[0] = SDI_WRITE_MEMORY;
+             store_long_parameter (buf + 1, 0xffff8080 + 4 * i);
+             store_long_parameter (buf + 5, 4);
+             store_unsigned_integer (buf + 9, 4, bp_addr);
+             send_data (buf, 13);
+           }
+       }
+
+      /* Set dbt breakpoints. */
+      for (i = ib_breakpoints; i < MAX_BREAKPOINTS; i++)
+       {
+         bp_addr = bp_address[i];
+         if (bp_addr != 0xffffffff && bp_addr != pc_addr)
+           {
+             if (!mmu_on)
+               bp_addr &= 0x7fffffff;
+
+             /* Write DBT instruction. */
+             buf[0] = SDI_WRITE_MEMORY;
+             if ((bp_addr & 2) == 0 && bp_addr != (pc_addr & 0xfffffffc))
+               {
+                 store_long_parameter (buf + 1, bp_addr);
+                 store_long_parameter (buf + 5, 4);
+                 if (TARGET_BYTE_ORDER == BFD_ENDIAN_BIG)
+                   {
+                     buf[9] = dbt_bp_entry[0];
+                     buf[10] = dbt_bp_entry[1];
+                     buf[11] = dbt_bp_entry[2];
+                     buf[12] = dbt_bp_entry[3];
+                   }
+                 else
+                   {
+                     buf[9] = dbt_bp_entry[3];
+                     buf[10] = dbt_bp_entry[2];
+                     buf[11] = dbt_bp_entry[1];
+                     buf[12] = dbt_bp_entry[0];
+                   }
+                 send_data (buf, 13);
+               }
+             else
+               {
+                 if (TARGET_BYTE_ORDER == BFD_ENDIAN_BIG)
+                   store_long_parameter (buf + 1, bp_addr);
+                 else if ((bp_addr & 2) == 0)
+                   store_long_parameter (buf + 1, bp_addr + 2);
+                 else
+                   store_long_parameter (buf + 1, bp_addr - 2);
+                 store_long_parameter (buf + 5, 2);
+                 if (TARGET_BYTE_ORDER == BFD_ENDIAN_BIG)
+                   {
+                     buf[9] = dbt_bp_entry[0];
+                     buf[10] = dbt_bp_entry[1];
+                   }
+                 else
+                   {
+                     buf[9] = dbt_bp_entry[1];
+                     buf[10] = dbt_bp_entry[0];
+                   }
+                 send_data (buf, 11);
+               }
+           }
+       }
+
+      /* Set access breaks. */
+      for (i = 0; i < max_access_breaks; i++)
+       {
+         ab_addr = ab_address[i];
+         if (ab_addr != 0x00000000)
+           {
+             /* DBC register */
+             buf[0] = SDI_WRITE_MEMORY;
+             store_long_parameter (buf + 1, 0xffff8100 + 4 * i);
+             store_long_parameter (buf + 5, 4);
+             if (TARGET_BYTE_ORDER == BFD_ENDIAN_BIG)
+               {
+                 buf[9] = 0x00;
+                 buf[10] = 0x00;
+                 buf[11] = 0x00;
+                 switch (ab_type[i])
+                   {
+                   case 0:     /* write watch */
+                     buf[12] = 0x86;
+                     break;
+                   case 1:     /* read watch */
+                     buf[12] = 0x46;
+                     break;
+                   case 2:     /* access watch */
+                     buf[12] = 0x06;
+                     break;
+                   }
+               }
+             else
+               {
+                 switch (ab_type[i])
+                   {
+                   case 0:     /* write watch */
+                     buf[9] = 0x86;
+                     break;
+                   case 1:     /* read watch */
+                     buf[9] = 0x46;
+                     break;
+                   case 2:     /* access watch */
+                     buf[9] = 0x06;
+                     break;
+                   }
+                 buf[10] = 0x00;
+                 buf[11] = 0x00;
+                 buf[12] = 0x00;
+               }
+             send_data (buf, 13);
+
+             /* DBAH register */
+             buf[0] = SDI_WRITE_MEMORY;
+             store_long_parameter (buf + 1, 0xffff8180 + 4 * i);
+             store_long_parameter (buf + 5, 4);
+             store_unsigned_integer (buf + 9, 4, ab_addr);
+             send_data (buf, 13);
+
+             /* DBAL register */
+             buf[0] = SDI_WRITE_MEMORY;
+             store_long_parameter (buf + 1, 0xffff8200 + 4 * i);
+             store_long_parameter (buf + 5, 4);
+             store_long_parameter (buf + 9, 0xffffffff);
+             send_data (buf, 13);
+
+             /* DBD register */
+             buf[0] = SDI_WRITE_MEMORY;
+             store_long_parameter (buf + 1, 0xffff8280 + 4 * i);
+             store_long_parameter (buf + 5, 4);
+             store_long_parameter (buf + 9, 0x00000000);
+             send_data (buf, 13);
+
+             /* DBDM register */
+             buf[0] = SDI_WRITE_MEMORY;
+             store_long_parameter (buf + 1, 0xffff8300 + 4 * i);
+             store_long_parameter (buf + 5, 4);
+             store_long_parameter (buf + 9, 0x00000000);
+             send_data (buf, 13);
+           }
+       }
+
+      /* Unset PBP. */
+      buf[0] = SDI_WRITE_CPU_REG;
+      buf[1] = SDI_REG_PBP;
+      store_long_parameter (buf + 2, 0x00000000);
+      send_data (buf, 6);
+    }
+
+  buf[0] = SDI_EXEC_CPU;
+  send_data (buf, 1);
+
+  /* Without this, some commands which require an active target (such as kill)
+     won't work.  This variable serves (at least) double duty as both the pid
+     of the target process (if it has such), and as a flag indicating that a
+     target is active.  These functions should be split out into seperate
+     variables, especially since GDB will someday have a notion of debugging
+     several processes.  */
+  inferior_ptid = pid_to_ptid (32);
+
+  return;
+}
+
+/* Wait until the remote machine stops, then return,
+   storing status in STATUS just as `wait' would.  */
+
+static void
+gdb_cntrl_c (int signo)
+{
+  if (remote_debug)
+    fprintf_unfiltered (gdb_stdlog, "interrupt\n");
+  interrupted = 1;
+}
+
+static ptid_t
+m32r_wait (ptid_t ptid, struct target_waitstatus *status)
+{
+  static RETSIGTYPE (*prev_sigint) ();
+  unsigned long bp_addr, pc_addr;
+  long i;
+  unsigned char buf[13];
+  unsigned long val;
+  int ret, c;
+
+  if (remote_debug)
+    fprintf_unfiltered (gdb_stdlog, "m32r_wait()\n");
+
+  status->kind = TARGET_WAITKIND_EXITED;
+  status->value.sig = 0;
+
+  interrupted = 0;
+  prev_sigint = signal (SIGINT, gdb_cntrl_c);
+
+  /* Wait for ready */
+  buf[0] = SDI_WAIT_FOR_READY;
+  if (serial_write (sdi_desc, buf, 1) != 0)
+    error ("Remote connection closed");
+
+  while (1)
+    {
+      c = serial_readchar (sdi_desc, SDI_TIMEOUT);
+      if (c < 0)
+       error ("Remote connection closed");
+
+      if (c == '-')    /* error */
+       {
+         status->kind = TARGET_WAITKIND_STOPPED;
+         status->value.sig = TARGET_SIGNAL_HUP;
+         return inferior_ptid;
+       }
+      else if (c == '+')       /* stopped */
+       break;
+
+      if (interrupted)
+       ret = serial_write (sdi_desc, "!", 1);  /* packet to interrupt */
+      else
+       ret = serial_write (sdi_desc, ".", 1);  /* packet to wait */
+      if (ret != 0)
+       error ("Remote connection closed");
+    }
+
+  status->kind = TARGET_WAITKIND_STOPPED;
+  if (interrupted)
+    status->value.sig = TARGET_SIGNAL_INT;
+  else
+    status->value.sig = TARGET_SIGNAL_TRAP;
+
+  interrupted = 0;
+  signal (SIGINT, prev_sigint);
+
+  check_mmu_status ();
+
+  /* Recover parallel bit. */
+  if (last_pc_addr != 0xffffffff)
+    {
+      buf[0] = SDI_WRITE_MEMORY;
+      if (TARGET_BYTE_ORDER == BFD_ENDIAN_BIG)
+       store_long_parameter (buf + 1, last_pc_addr);
+      else
+       store_long_parameter (buf + 1, last_pc_addr - 1);
+      store_long_parameter (buf + 5, 1);
+      buf[9] = last_pc_addr_data[0];
+      send_data (buf, 10);
+      last_pc_addr = 0xffffffff;
+    }
+
+  /* Breakpoints are inserted only for "next" command */
+  if (!step_mode)
+    {
+      int ib_breakpoints;
+
+      if (use_ib_breakpoints)
+       ib_breakpoints = max_ib_breakpoints;
+      else
+       ib_breakpoints = 0;
+
+      /* Set back pc by 2 if m32r is stopped with dbt. */
+      buf[0] = SDI_READ_CPU_REG;
+      buf[1] = SDI_REG_BPC;
+      send_data (buf, 2);
+      recv_data (&val, 4);
+      pc_addr = ntohl (val) - 2;
+      for (i = ib_breakpoints; i < MAX_BREAKPOINTS; i++)
+       {
+         if (pc_addr == bp_address[i])
+           {
+             buf[0] = SDI_WRITE_CPU_REG;
+             buf[1] = SDI_REG_BPC;
+             store_long_parameter (buf + 2, pc_addr);
+             send_data (buf, 6);
+
+             /* If there is a parallel instruction with +2 offset at pc
+                address, we have to take care of it later. */
+             if ((pc_addr & 0x2) != 0)
+               {
+                 if (TARGET_BYTE_ORDER == BFD_ENDIAN_BIG)
+                   {
+                     if ((bp_data[i][2] & 0x80) != 0)
+                       {
+                         last_pc_addr = pc_addr;
+                         last_pc_addr_data[0] = bp_data[i][2];
+                         last_pc_addr_data[1] = bp_data[i][3];
+                       }
+                   }
+                 else
+                   {
+                     if ((bp_data[i][1] & 0x80) != 0)
+                       {
+                         last_pc_addr = pc_addr;
+                         last_pc_addr_data[0] = bp_data[i][1];
+                         last_pc_addr_data[1] = bp_data[i][0];
+                       }
+                   }
+               }
+             break;
+           }
+       }
+
+      /* Remove ib breakpoints. */
+      for (i = 0; i < ib_breakpoints; i++)
+       {
+         if (bp_address[i] != 0xffffffff)
+           {
+             buf[0] = SDI_WRITE_MEMORY;
+             store_long_parameter (buf + 1, 0xffff8000 + 4 * i);
+             store_long_parameter (buf + 5, 4);
+             buf[9] = ib_bp_entry_disable[0];
+             buf[10] = ib_bp_entry_disable[1];
+             buf[11] = ib_bp_entry_disable[2];
+             buf[12] = ib_bp_entry_disable[3];
+             send_data (buf, 13);
+           }
+       }
+      /* Remove dbt breakpoints. */
+      for (i = ib_breakpoints; i < MAX_BREAKPOINTS; i++)
+       {
+         bp_addr = bp_address[i];
+         if (bp_addr != 0xffffffff)
+           {
+             if (!mmu_on)
+               bp_addr &= 0x7fffffff;
+             buf[0] = SDI_WRITE_MEMORY;
+             store_long_parameter (buf + 1, bp_addr & 0xfffffffc);
+             store_long_parameter (buf + 5, 4);
+             buf[9] = bp_data[i][0];
+             buf[10] = bp_data[i][1];
+             buf[11] = bp_data[i][2];
+             buf[12] = bp_data[i][3];
+             send_data (buf, 13);
+           }
+       }
+
+      /* Remove access breaks. */
+      hit_watchpoint_addr = 0;
+      for (i = 0; i < max_access_breaks; i++)
+       {
+         if (ab_address[i] != 0x00000000)
+           {
+             buf[0] = SDI_READ_MEMORY;
+             store_long_parameter (buf + 1, 0xffff8100 + 4 * i);
+             store_long_parameter (buf + 5, 4);
+             serial_write (sdi_desc, buf, 9);
+             c = serial_readchar (sdi_desc, SDI_TIMEOUT);
+             if (c != '-' && recv_data (buf, 4) != -1)
+               {
+                 if (TARGET_BYTE_ORDER == BFD_ENDIAN_BIG)
+                   {
+                     if ((buf[3] & 0x1) == 0x1)
+                       hit_watchpoint_addr = ab_address[i];
+                   }
+                 else
+                   {
+                     if ((buf[0] & 0x1) == 0x1)
+                       hit_watchpoint_addr = ab_address[i];
+                   }
+               }
+
+             buf[0] = SDI_WRITE_MEMORY;
+             store_long_parameter (buf + 1, 0xffff8100 + 4 * i);
+             store_long_parameter (buf + 5, 4);
+             store_long_parameter (buf + 9, 0x00000000);
+             send_data (buf, 13);
+           }
+       }
+
+      if (remote_debug)
+       fprintf_unfiltered (gdb_stdlog, "pc => 0x%lx\n", pc_addr);
+    }
+  else
+    last_pc_addr = 0xffffffff;
+
+  return inferior_ptid;
+}
+
+/* Terminate the open connection to the remote debugger.
+   Use this when you want to detach and do something else
+   with your gdb.  */
+static void
+m32r_detach (char *args, int from_tty)
+{
+  if (remote_debug)
+    fprintf_unfiltered (gdb_stdlog, "m32r_detach(%d)\n", from_tty);
+
+  m32r_resume (inferior_ptid, 0, 0);
+
+  /* calls m32r_close to do the real work */
+  pop_target ();
+  if (from_tty)
+    fprintf_unfiltered (gdb_stdlog, "Ending remote %s debugging\n",
+                       target_shortname);
+}
+
+/* Return the id of register number REGNO. */
+
+static int
+get_reg_id (int regno)
+{
+  switch (regno)
+    {
+    case 20:
+      return SDI_REG_BBPC;
+    case 21:
+      return SDI_REG_BPC;
+    case 22:
+      return SDI_REG_ACCL;
+    case 23:
+      return SDI_REG_ACCH;
+    case 24:
+      return SDI_REG_EVB;
+    }
+
+  return regno;
+}
+
+/* Read the remote registers into the block REGS.  */
+
+static void m32r_fetch_register (int);
+
+static void
+m32r_fetch_registers (void)
+{
+  int regno;
+
+  for (regno = 0; regno < NUM_REGS; regno++)
+    m32r_fetch_register (regno);
+}
+
+/* Fetch register REGNO, or all registers if REGNO is -1.
+   Returns errno value.  */
+static void
+m32r_fetch_register (int regno)
+{
+  unsigned long val, val2, regid;
+  unsigned char buf[2];
+
+  if (regno == -1)
+    m32r_fetch_registers ();
+  else
+    {
+      char buffer[MAX_REGISTER_SIZE];
+
+      regid = get_reg_id (regno);
+      buf[0] = SDI_READ_CPU_REG;
+      buf[1] = regid;
+      send_data (buf, 2);
+      recv_data (&val, 4);
+      val = ntohl (val);
+
+      if (regid == SDI_REG_PSW)
+       {
+         buf[0] = SDI_READ_CPU_REG;
+         buf[1] = SDI_REG_BBPSW;
+         send_data (buf, 2);
+         recv_data (&val2, 4);
+         val2 = ntohl (val2);
+         val = ((0x00c1 & val2) << 8) | ((0xc100 & val) >> 8);
+       }
+
+      if (remote_debug)
+       fprintf_unfiltered (gdb_stdlog, "m32r_fetch_register(%d,0x%08lx)\n",
+                           regno, val);
+
+      /* We got the number the register holds, but gdb expects to see a
+         value in the target byte ordering.  */
+      store_unsigned_integer (buffer, 4, val);
+      supply_register (regno, buffer);
+    }
+  return;
+}
+
+/* Store the remote registers from the contents of the block REGS.  */
+
+static void m32r_store_register (int);
+
+static void
+m32r_store_registers (void)
+{
+  int regno;
+
+  for (regno = 0; regno < NUM_REGS; regno++)
+    m32r_store_register (regno);
+
+  registers_changed ();
+}
+
+/* Store register REGNO, or all if REGNO == 0.
+   Return errno value.  */
+static void
+m32r_store_register (int regno)
+{
+  int regid;
+  ULONGEST regval, tmp;
+  unsigned char buf[6];
+
+  if (regno == -1)
+    m32r_store_registers ();
+  else
+    {
+      regcache_cooked_read_unsigned (current_regcache, regno, &regval);
+      regid = get_reg_id (regno);
+
+      if (regid == SDI_REG_PSW)
+       {
+         unsigned long psw, bbpsw;
+
+         buf[0] = SDI_READ_CPU_REG;
+         buf[1] = SDI_REG_PSW;
+         send_data (buf, 2);
+         recv_data (&psw, 4);
+         psw = ntohl (psw);
+
+         buf[0] = SDI_READ_CPU_REG;
+         buf[1] = SDI_REG_BBPSW;
+         send_data (buf, 2);
+         recv_data (&bbpsw, 4);
+         bbpsw = ntohl (bbpsw);
+
+         tmp = (0x00c1 & psw) | ((0x00c1 & regval) << 8);
+         buf[0] = SDI_WRITE_CPU_REG;
+         buf[1] = SDI_REG_PSW;
+         store_long_parameter (buf + 2, tmp);
+         send_data (buf, 6);
+
+         tmp = (0x0030 & bbpsw) | ((0xc100 & regval) >> 8);
+         buf[0] = SDI_WRITE_CPU_REG;
+         buf[1] = SDI_REG_BBPSW;
+         store_long_parameter (buf + 2, tmp);
+         send_data (buf, 6);
+       }
+      else
+       {
+         buf[0] = SDI_WRITE_CPU_REG;
+         buf[1] = regid;
+         store_long_parameter (buf + 2, regval);
+         send_data (buf, 6);
+       }
+
+      if (remote_debug)
+       fprintf_unfiltered (gdb_stdlog, "m32r_store_register(%d,0x%08lu)\n",
+                           regno, (unsigned long) regval);
+    }
+}
+
+/* Get ready to modify the registers array.  On machines which store
+   individual registers, this doesn't need to do anything.  On machines
+   which store all the registers in one fell swoop, this makes sure
+   that registers contains all the registers from the program being
+   debugged.  */
+
+static void
+m32r_prepare_to_store (void)
+{
+  /* Do nothing, since we can store individual regs */
+  if (remote_debug)
+    fprintf_unfiltered (gdb_stdlog, "m32r_prepare_to_store()\n");
+}
+
+static void
+m32r_files_info (struct target_ops *target)
+{
+  char *file = "nothing";
+
+  if (exec_bfd)
+    {
+      file = bfd_get_filename (exec_bfd);
+      printf_filtered ("\tAttached to %s running program %s\n",
+                      chip_name, file);
+    }
+}
+
+/* Read/Write memory.  */
+static int
+m32r_xfer_memory (CORE_ADDR memaddr, char *myaddr, int len,
+                 int write,
+                 struct mem_attrib *attrib, struct target_ops *target)
+{
+  unsigned long taddr;
+  unsigned char buf[0x2000];
+  int ret, c;
+
+  taddr = memaddr;
+
+  if (!mmu_on)
+    {
+      if ((taddr & 0xa0000000) == 0x80000000)
+       taddr &= 0x7fffffff;
+    }
+
+  if (remote_debug)
+    {
+      if (write)
+       fprintf_unfiltered (gdb_stdlog, "m32r_xfer_memory(%08lx,%d,write)\n",
+                           memaddr, len);
+      else
+       fprintf_unfiltered (gdb_stdlog, "m32r_xfer_memory(%08lx,%d,read)\n",
+                           memaddr, len);
+    }
+
+  if (write)
+    {
+      buf[0] = SDI_WRITE_MEMORY;
+      store_long_parameter (buf + 1, taddr);
+      store_long_parameter (buf + 5, len);
+      if (len < 0x1000)
+       {
+         memcpy (buf + 9, myaddr, len);
+         ret = send_data (buf, len + 9) - 9;
+       }
+      else
+       {
+         if (serial_write (sdi_desc, buf, 9) != 0)
+           {
+             if (remote_debug)
+               fprintf_unfiltered (gdb_stdlog,
+                                   "m32r_xfer_memory() failed\n");
+             return 0;
+           }
+         ret = send_data (myaddr, len);
+       }
+    }
+  else
+    {
+      buf[0] = SDI_READ_MEMORY;
+      store_long_parameter (buf + 1, taddr);
+      store_long_parameter (buf + 5, len);
+      if (serial_write (sdi_desc, buf, 9) != 0)
+       {
+         if (remote_debug)
+           fprintf_unfiltered (gdb_stdlog, "m32r_xfer_memory() failed\n");
+         return 0;
+       }
+
+      c = serial_readchar (sdi_desc, SDI_TIMEOUT);
+      if (c < 0 || c == '-')
+       {
+         if (remote_debug)
+           fprintf_unfiltered (gdb_stdlog, "m32r_xfer_memory() failed\n");
+         return 0;
+       }
+
+      ret = recv_data (myaddr, len);
+    }
+
+  if (ret <= 0)
+    {
+      if (remote_debug)
+       fprintf_unfiltered (gdb_stdlog, "m32r_xfer_memory() fails\n");
+      return 0;
+    }
+
+  return ret;
+}
+
+static void
+m32r_kill (void)
+{
+  if (remote_debug)
+    fprintf_unfiltered (gdb_stdlog, "m32r_kill()\n");
+
+  inferior_ptid = null_ptid;
+
+  return;
+}
+
+/* Clean up when a program exits.
+
+   The program actually lives on in the remote processor's RAM, and may be
+   run again without a download.  Don't leave it full of breakpoint
+   instructions.  */
+
+static void
+m32r_mourn_inferior (void)
+{
+  if (remote_debug)
+    fprintf_unfiltered (gdb_stdlog, "m32r_mourn_inferior()\n");
+
+  remove_breakpoints ();
+  generic_mourn_inferior ();
+}
+
+static int
+m32r_insert_breakpoint (CORE_ADDR addr, char *shadow)
+{
+  int ib_breakpoints;
+  unsigned char buf[13];
+  int i, c;
+
+  if (remote_debug)
+    fprintf_unfiltered (gdb_stdlog, "m32r_insert_breakpoint(%08lx,\"%s\")\n",
+                       addr, shadow);
+
+  if (use_ib_breakpoints)
+    ib_breakpoints = max_ib_breakpoints;
+  else
+    ib_breakpoints = 0;
+
+  for (i = 0; i < MAX_BREAKPOINTS; i++)
+    {
+      if (bp_address[i] == 0xffffffff)
+       {
+         bp_address[i] = addr;
+         if (i >= ib_breakpoints)
+           {
+             buf[0] = SDI_READ_MEMORY;
+             if (mmu_on)
+               store_long_parameter (buf + 1, addr & 0xfffffffc);
+             else
+               store_long_parameter (buf + 1, addr & 0x7ffffffc);
+             store_long_parameter (buf + 5, 4);
+             serial_write (sdi_desc, buf, 9);
+             c = serial_readchar (sdi_desc, SDI_TIMEOUT);
+             if (c != '-')
+               recv_data (bp_data[i], 4);
+           }
+         return 0;
+       }
+    }
+
+  error ("Too many breakpoints");
+  return 1;
+}
+
+static int
+m32r_remove_breakpoint (CORE_ADDR addr, char *shadow)
+{
+  int i;
+
+  if (remote_debug)
+    fprintf_unfiltered (gdb_stdlog, "m32r_remove_breakpoint(%08lx,\"%s\")\n",
+                       addr, shadow);
+
+  for (i = 0; i < MAX_BREAKPOINTS; i++)
+    {
+      if (bp_address[i] == addr)
+       {
+         bp_address[i] = 0xffffffff;
+         break;
+       }
+    }
+
+  return 0;
+}
+
+static void
+m32r_load (char *args, int from_tty)
+{
+  struct cleanup *old_chain;
+  asection *section;
+  bfd *pbfd;
+  bfd_vma entry;
+  char *filename;
+  int quiet;
+  int nostart;
+  time_t start_time, end_time; /* Start and end times of download */
+  unsigned long data_count;    /* Number of bytes transferred to memory */
+  int ret;
+  static RETSIGTYPE (*prev_sigint) ();
+
+  /* for direct tcp connections, we can do a fast binary download */
+  quiet = 0;
+  nostart = 0;
+  filename = NULL;
+
+  while (*args != '\000')
+    {
+      char *arg;
+
+      while (isspace (*args))
+       args++;
+
+      arg = args;
+
+      while ((*args != '\000') && !isspace (*args))
+       args++;
+
+      if (*args != '\000')
+       *args++ = '\000';
+
+      if (*arg != '-')
+       filename = arg;
+      else if (strncmp (arg, "-quiet", strlen (arg)) == 0)
+       quiet = 1;
+      else if (strncmp (arg, "-nostart", strlen (arg)) == 0)
+       nostart = 1;
+      else
+       error ("Unknown option `%s'", arg);
+    }
+
+  if (!filename)
+    filename = get_exec_file (1);
+
+  pbfd = bfd_openr (filename, gnutarget);
+  if (pbfd == NULL)
+    {
+      perror_with_name (filename);
+      return;
+    }
+  old_chain = make_cleanup_bfd_close (pbfd);
+
+  if (!bfd_check_format (pbfd, bfd_object))
+    error ("\"%s\" is not an object file: %s", filename,
+          bfd_errmsg (bfd_get_error ()));
+
+  start_time = time (NULL);
+  data_count = 0;
+
+  interrupted = 0;
+  prev_sigint = signal (SIGINT, gdb_cntrl_c);
+
+  for (section = pbfd->sections; section; section = section->next)
+    {
+      if (bfd_get_section_flags (pbfd, section) & SEC_LOAD)
+       {
+         bfd_vma section_address;
+         bfd_size_type section_size;
+         file_ptr fptr;
+         int n;
+
+         section_address = bfd_section_lma (pbfd, section);
+         section_size = bfd_get_section_size_before_reloc (section);
+
+         if (!mmu_on)
+           {
+             if ((section_address & 0xa0000000) == 0x80000000)
+               section_address &= 0x7fffffff;
+           }
+
+         if (!quiet)
+           printf_filtered ("[Loading section %s at 0x%lx (%d bytes)]\n",
+                            bfd_get_section_name (pbfd, section),
+                            section_address, (int) section_size);
+
+         fptr = 0;
+
+         data_count += section_size;
+
+         n = 0;
+         while (section_size > 0)
+           {
+             char unsigned buf[0x1000 + 9];
+             int count;
+
+             count = min (section_size, 0x1000);
+
+             buf[0] = SDI_WRITE_MEMORY;
+             store_long_parameter (buf + 1, section_address);
+             store_long_parameter (buf + 5, count);
+
+             bfd_get_section_contents (pbfd, section, buf + 9, fptr, count);
+             if (send_data (buf, count + 9) <= 0)
+               error ("Error while downloading %s section.",
+                      bfd_get_section_name (pbfd, section));
+
+             if (!quiet)
+               {
+                 printf_unfiltered (".");
+                 if (n++ > 60)
+                   {
+                     printf_unfiltered ("\n");
+                     n = 0;
+                   }
+                 gdb_flush (gdb_stdout);
+               }
+
+             section_address += count;
+             fptr += count;
+             section_size -= count;
+
+             if (interrupted)
+               break;
+           }
+
+         if (!quiet && !interrupted)
+           {
+             printf_unfiltered ("done.\n");
+             gdb_flush (gdb_stdout);
+           }
+       }
+
+      if (interrupted)
+       {
+         printf_unfiltered ("Interrupted.\n");
+         break;
+       }
+    }
+
+  interrupted = 0;
+  signal (SIGINT, prev_sigint);
+
+  end_time = time (NULL);
+
+  /* Make the PC point at the start address */
+  if (exec_bfd)
+    write_pc (bfd_get_start_address (exec_bfd));
+
+  inferior_ptid = null_ptid;   /* No process now */
+
+  /* This is necessary because many things were based on the PC at the time
+     that we attached to the monitor, which is no longer valid now that we
+     have loaded new code (and just changed the PC).  Another way to do this
+     might be to call normal_stop, except that the stack may not be valid,
+     and things would get horribly confused... */
+
+  clear_symtab_users ();
+
+  if (!nostart)
+    {
+      entry = bfd_get_start_address (pbfd);
+
+      if (!quiet)
+       printf_unfiltered ("[Starting %s at 0x%lx]\n", filename, entry);
+    }
+
+  print_transfer_performance (gdb_stdout, data_count, 0,
+                             end_time - start_time);
+
+  do_cleanups (old_chain);
+}
+
+static void
+m32r_stop (void)
+{
+  unsigned char buf[1];
+
+  if (remote_debug)
+    fprintf_unfiltered (gdb_stdlog, "m32r_stop()\n");
+
+  buf[0] = SDI_STOP_CPU;
+  send_data (buf, 1);
+
+  return;
+}
+
+
+/* Tell whether this target can support a hardware breakpoint. 
+   This implements the TARGET_CAN_USE_HARDWARE_WATCHPOINT macro.  */
+
+int
+m32r_can_use_hardware_watchpoint (void)
+{
+  return max_access_breaks;
+}
+
+/* Set a data watchpoint.  ADDR and LEN should be obvious.  TYPE is 0
+   for a write watchpoint, 1 for a read watchpoint, or 2 for a read/write
+   watchpoint. */
+
+int
+m32r_set_watchpoint (CORE_ADDR addr, int len, int type)
+{
+  int i;
+
+  if (remote_debug)
+    fprintf_unfiltered (gdb_stdlog, "m32r_set_watchpoint(%08lx,%d,%d)\n",
+                       addr, len, type);
+
+  for (i = 0; i < MAX_ACCESS_BREAKS; i++)
+    {
+      if (ab_address[i] == 0x00000000)
+       {
+         ab_address[i] = addr;
+         ab_size[i] = len;
+         ab_type[i] = type;
+         return 0;
+       }
+    }
+
+  error ("Too many watchpoints");
+  return 1;
+}
+
+int
+m32r_remove_watchpoint (CORE_ADDR addr, int len, int type)
+{
+  int i;
+
+  if (remote_debug)
+    fprintf_unfiltered (gdb_stdlog, "m32r_remove_watchpoint(%08lx,%d,%d)\n",
+                       addr, len, type);
+
+  for (i = 0; i < MAX_ACCESS_BREAKS; i++)
+    {
+      if (ab_address[i] == addr)
+       {
+         ab_address[i] = 0x00000000;
+         break;
+       }
+    }
+
+  return 0;
+}
+
+CORE_ADDR
+m32r_stopped_data_address (void)
+{
+  return hit_watchpoint_addr;
+}
+
+int
+m32r_stopped_by_watchpoint (void)
+{
+  return (hit_watchpoint_addr != 0x00000000);
+}
+
+
+static void
+sdireset_command (char *args, int from_tty)
+{
+  unsigned char buf[1];
+
+  if (remote_debug)
+    fprintf_unfiltered (gdb_stdlog, "m32r_sdireset()\n");
+
+  buf[0] = SDI_OPEN;
+  send_data (buf, 1);
+
+  inferior_ptid = null_ptid;
+}
+
+
+static void
+sdistatus_command (char *args, int from_tty)
+{
+  unsigned char buf[4096];
+  int i, c;
+
+  if (remote_debug)
+    fprintf_unfiltered (gdb_stdlog, "m32r_sdireset()\n");
+
+  if (!sdi_desc)
+    return;
+
+  buf[0] = SDI_STATUS;
+  send_data (buf, 1);
+  for (i = 0; i < 4096; i++)
+    {
+      c = serial_readchar (sdi_desc, SDI_TIMEOUT);
+      if (c < 0)
+        return;
+      buf[i] = c;
+      if (c == 0)
+        break;
+    }    
+
+  printf_filtered ("%s", buf);
+}
+
+
+static void
+debug_chaos_command (char *args, int from_tty)
+{
+  unsigned char buf[3];
+
+  buf[0] = SDI_SET_ATTR;
+  buf[1] = SDI_ATTR_CACHE;
+  buf[2] = SDI_CACHE_TYPE_CHAOS;
+  send_data (buf, 3);
+}
+
+
+static void
+use_debug_dma_command (char *args, int from_tty)
+{
+  unsigned char buf[3];
+
+  buf[0] = SDI_SET_ATTR;
+  buf[1] = SDI_ATTR_MEM_ACCESS;
+  buf[2] = SDI_MEM_ACCESS_DEBUG_DMA;
+  send_data (buf, 3);
+}
+
+static void
+use_mon_code_command (char *args, int from_tty)
+{
+  unsigned char buf[3];
+
+  buf[0] = SDI_SET_ATTR;
+  buf[1] = SDI_ATTR_MEM_ACCESS;
+  buf[2] = SDI_MEM_ACCESS_MON_CODE;
+  send_data (buf, 3);
+}
+
+
+static void
+use_ib_breakpoints_command (char *args, int from_tty)
+{
+  use_ib_breakpoints = 1;
+}
+
+static void
+use_dbt_breakpoints_command (char *args, int from_tty)
+{
+  use_ib_breakpoints = 0;
+}
+
+
+/* Define the target subroutine names */
+
+struct target_ops m32r_ops;
+
+static void
+init_m32r_ops (void)
+{
+  m32r_ops.to_shortname = "m32rsdi";
+  m32r_ops.to_longname = "Remote M32R debugging over SDI interface";
+  m32r_ops.to_doc = "Use an M32R board using SDI debugging protocol.";
+  m32r_ops.to_open = m32r_open;
+  m32r_ops.to_close = m32r_close;
+  m32r_ops.to_detach = m32r_detach;
+  m32r_ops.to_resume = m32r_resume;
+  m32r_ops.to_wait = m32r_wait;
+  m32r_ops.to_fetch_registers = m32r_fetch_register;
+  m32r_ops.to_store_registers = m32r_store_register;
+  m32r_ops.to_prepare_to_store = m32r_prepare_to_store;
+  m32r_ops.to_xfer_memory = m32r_xfer_memory;
+  m32r_ops.to_files_info = m32r_files_info;
+  m32r_ops.to_insert_breakpoint = m32r_insert_breakpoint;
+  m32r_ops.to_remove_breakpoint = m32r_remove_breakpoint;
+  m32r_ops.to_kill = m32r_kill;
+  m32r_ops.to_load = m32r_load;
+  m32r_ops.to_create_inferior = m32r_create_inferior;
+  m32r_ops.to_mourn_inferior = m32r_mourn_inferior;
+  m32r_ops.to_stop = m32r_stop;
+  m32r_ops.to_stratum = process_stratum;
+  m32r_ops.to_has_all_memory = 1;
+  m32r_ops.to_has_memory = 1;
+  m32r_ops.to_has_stack = 1;
+  m32r_ops.to_has_registers = 1;
+  m32r_ops.to_has_execution = 1;
+  m32r_ops.to_magic = OPS_MAGIC;
+};
+
+
+extern initialize_file_ftype _initialize_remote_m32r;
+
+void
+_initialize_remote_m32r (void)
+{
+  int i;
+
+  init_m32r_ops ();
+
+  /* Initialize breakpoints. */
+  for (i = 0; i < MAX_BREAKPOINTS; i++)
+    bp_address[i] = 0xffffffff;
+
+  /* Initialize access breaks. */
+  for (i = 0; i < MAX_ACCESS_BREAKS; i++)
+    ab_address[i] = 0x00000000;
+
+  add_target (&m32r_ops);
+
+  add_com ("sdireset", class_obscure, sdireset_command,
+          "Reset SDI connection.");
+
+  add_com ("sdistatus", class_obscure, sdistatus_command,
+          "Show status of SDI connection.");
+
+  add_com ("debug_chaos", class_obscure, debug_chaos_command,
+          "Debug M32R/Chaos.");
+
+  add_com ("use_debug_dma", class_obscure, use_debug_dma_command,
+          "Use debug DMA mem access.");
+  add_com ("use_mon_code", class_obscure, use_mon_code_command,
+          "Use mon code mem access.");
+
+  add_com ("use_ib_break", class_obscure, use_ib_breakpoints_command,
+          "Set breakpoints by IB break.");
+  add_com ("use_dbt_break", class_obscure, use_dbt_breakpoints_command,
+          "Set breakpoints by dbt.");
+}
diff --git a/gdb/testsuite/gdb.arch/gdb1291.c b/gdb/testsuite/gdb.arch/gdb1291.c
new file mode 100755 (executable)
index 0000000..2178f70
--- /dev/null
@@ -0,0 +1,44 @@
+/* Copyright 2003 Free Software Foundation, Inc.\r
\r
+   This program is free software; you can redistribute it and/or modify\r
+   it under the terms of the GNU General Public License as published by\r
+   the Free Software Foundation; either version 2 of the License, or\r
+   (at your option) any later version.\r
\r
+   This program is distributed in the hope that it will be useful,\r
+   but WITHOUT ANY WARRANTY; without even the implied warranty of\r
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\r
+   GNU General Public License for more details.\r
\r
+   You should have received a copy of the GNU General Public License\r
+   along with this program; if not, write to the Free Software\r
+   Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.  \r
\r
+   Please email any bugs, comments, and/or additions to this file to:\r
+   bug-gdb@gnu.org\r
\r
+   This file is part of the gdb testsuite.  */\r
+\r
+void sub (void);\r
+\r
+main()\r
+{\r
+        sub();\r
+}\r
+\r
+asm(".text\n"\r
+    "    .align 5\n"\r
+    "sub:\n"\r
+    "    mov.l  r14,@-r15\n"\r
+    "    mov.w  .STACK2,r3\n"\r
+    "    sub    r3,r15\n"\r
+    "    mov    r15,r14\n"\r
+    "    mov.w  .STACK2,r7\n"\r
+    "    add    r7,r14\n"\r
+    "    mov    r14,r15\n"\r
+    "    mov.l  @r15+,r14\n"\r
+    "    rts\n"\r
+    "    nop\n"\r
+    "    .align 1\n"\r
+    ".STACK2:\n"\r
+    "    .short  260\n");\r
diff --git a/gdb/testsuite/gdb.arch/gdb1291.exp b/gdb/testsuite/gdb.arch/gdb1291.exp
new file mode 100644 (file)
index 0000000..2c1f4cc
--- /dev/null
@@ -0,0 +1,62 @@
+# Copyright 2003 Free Software Foundation, Inc.
+
+# 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.
+#
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.  
+
+# Please email any bugs, comments, and/or additions to this file to:
+# bug-gdb@gnu.org
+
+# This file is part of the gdb testsuite.
+
+# Tests for PR:1291.  Ensure that backtrace works properly for stack
+# frames greater than 256 bytes.
+
+if $tracelevel {
+    strace $tracelevel
+}
+
+# Test SH backtraces with >256 byte frame stack. (PR:1291)
+
+set prms_id 0
+set bug_id 0
+
+if ![istarget "sh-*-*"] then {
+    verbose "Skipping SH backtrace tests."
+    return
+}
+
+set testfile "gdb1291"
+set srcfile ${testfile}.c
+set binfile ${objdir}/${subdir}/${testfile}
+if { [gdb_compile "${srcdir}/${subdir}/${srcfile}" "${binfile}" executable {debug}] != "" } {
+    gdb_suppress_entire_file "Testcase compile failed, so all tests in this file will automatically fail."
+}
+
+gdb_exit
+gdb_start
+gdb_reinitialize_dir $srcdir/$subdir
+gdb_load ${binfile}
+
+#
+# Run to `main' where we begin our tests.
+#
+
+if ![runto_main] then {
+    gdb_suppress_tests
+}
+
+gdb_test "b sub" "Breakpoint 2*" "set breakpoint"
+gdb_test "c" "Breakpoint 2*" "get to sub"
+gdb_test "bt" "#0\[ \t\]*$hex \\(\\) at sh-bt.*\r\n#1\[ \t\]*$hex in main.*" \
+       "backtrace in gdb1291"
diff --git a/gdb/testsuite/gdb.arch/gdb1431.c b/gdb/testsuite/gdb.arch/gdb1431.c
new file mode 100755 (executable)
index 0000000..0041042
--- /dev/null
@@ -0,0 +1,63 @@
+/* Copyright 2003 Free Software Foundation, Inc.\r
\r
+   This program is free software; you can redistribute it and/or modify\r
+   it under the terms of the GNU General Public License as published by\r
+   the Free Software Foundation; either version 2 of the License, or\r
+   (at your option) any later version.\r
\r
+   This program is distributed in the hope that it will be useful,\r
+   but WITHOUT ANY WARRANTY; without even the implied warranty of\r
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\r
+   GNU General Public License for more details.\r
\r
+   You should have received a copy of the GNU General Public License\r
+   along with this program; if not, write to the Free Software\r
+   Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.  \r
\r
+   Please email any bugs, comments, and/or additions to this file to:\r
+   bug-gdb@gnu.org\r
\r
+   This file is part of the gdb testsuite.  */\r
+\r
+void sub1 (void);\r
+void sub2 (void);\r
+\r
+main()\r
+{\r
+        sub1();\r
+        sub2();\r
+}\r
+\r
+asm(".text\n"\r
+    "    .align 5\n"\r
+    "sub1:\n"\r
+    "    mov.l  r14,@-r15\n"\r
+    "    add    #-128,r15\n"\r
+    "    add    #-128,r15\n"\r
+    "    mov    r15,r14\n"\r
+    "    mov.w  .STACK1,r7\n"\r
+    "    add    r7,r14\n"\r
+    "    mov    r14,r15\n"\r
+    "    mov.l  @r15+,r14\n"\r
+    "    rts\n"\r
+    "    nop\n"\r
+    "    .align 1\n"\r
+    ".STACK1:\n"\r
+    "    .short  256\n");\r
+\r
+asm(".text\n"\r
+    "    .align 5\n"\r
+    "sub2:\n"\r
+    "    mov.l  r14,@-r15\n"\r
+    "    mov.w  .STACK2,r3\n"\r
+    "    sub    r3,r15\n"\r
+    "    mov    r15,r14\n"\r
+    "    mov.w  .STACK2,r7\n"\r
+    "    add    r7,r14\n"\r
+    "    mov    r14,r15\n"\r
+    "    mov.l  @r15+,r14\n"\r
+    "    rts\n"\r
+    "    nop\n"\r
+    "    .align 1\n"\r
+    ".STACK2:\n"\r
+    "    .short  260\n");\r
diff --git a/gdb/testsuite/gdb.arch/gdb1431.exp b/gdb/testsuite/gdb.arch/gdb1431.exp
new file mode 100644 (file)
index 0000000..15bff46
--- /dev/null
@@ -0,0 +1,66 @@
+# Copyright 2003 Free Software Foundation, Inc.
+
+# 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.
+#
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.  
+
+# Please email any bugs, comments, and/or additions to this file to:
+# bug-gdb@gnu.org
+
+# This file is part of the gdb testsuite.
+
+# Tests for PR:1431.  Catch gdb not continuing to second function properly.
+
+if $tracelevel {
+    strace $tracelevel
+}
+
+# Observe that the until command doesn't go all the way to sub2.
+
+set prms_id 0
+set bug_id 0
+
+if ![istarget "sh-*-*"] then {
+    verbose "Skipping SH backtrace tests."
+    return
+}
+
+set testfile "gdb1431"
+set srcfile ${testfile}.c
+set binfile ${objdir}/${subdir}/${testfile}
+if { [gdb_compile "${srcdir}/${subdir}/${srcfile}" "${binfile}" executable {debug}] != "" } {
+    gdb_suppress_entire_file "Testcase compile failed, so all tests in this file will automatically fail."
+}
+
+gdb_exit
+gdb_start
+gdb_reinitialize_dir $srcdir/$subdir
+gdb_load ${binfile}
+
+#
+# Run to `main' where we begin our tests.
+#
+
+if ![runto_main] then {
+    gdb_suppress_tests
+}
+
+gdb_test "u sub1" "sub1*" "get to sub1"
+gdb_test "bt" "#0\[ \t\]*$hex \\(\\) at sh-bt.*\r\n#1\[ \t\]*$hex in main.*" \
+       "backtrace in gdb1291"
+
+kfail "gdb/1431" "u sub2"
+# This is what we would expect to be able to do:
+#gdb_test "u sub2" "sub2*" "get to sub2"
+#gdb_test "bt" "#0\[ \t\]*$hex \\(\\) at sh-bt.*\r\n#1\[ \t\]*$hex in main.*" \
+#      "backtrace in gdb1291"
diff --git a/gdb/testsuite/gdb.base/gdb1056.exp b/gdb/testsuite/gdb.base/gdb1056.exp
new file mode 100644 (file)
index 0000000..623f837
--- /dev/null
@@ -0,0 +1,48 @@
+# Copyright 2003 Free Software Foundation, Inc.
+
+# 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.
+# 
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+# GNU General Public License for more details.
+# 
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.  
+
+# Test for PR gdb/1056.
+# 2003-10-18  Michael Chastain <mec@shout.net>
+
+if $tracelevel then {
+    strace $tracelevel
+}
+
+# test SIGFPE (such as division by 0) inside gdb itself
+
+set prms_id 0
+set bug_id 0
+
+gdb_start
+
+# When SIGFPE happens, the operating system may restart the
+# offending instruction after the signal handler returns,
+# rather than proceeding to the next instruction.  This happens
+# on i686-pc-linux-gnu with a linux kernel.  If gdb has a naive
+# signal handler that just returns, then it will restart the
+# broken instruction and gdb gets an endless stream of SIGFPE's
+# and makes no progress.
+#
+# On a broken gdb this test will just time out.
+
+gdb_test_multiple "print 1/0" "" {
+    -re ".*$gdb_prompt $" {
+       pass "print 1/0"
+    }
+    timeout {
+       kfail "gdb/1056" "print 1/0"
+    }
+}
diff --git a/gdb/testsuite/gdb.cp/gdb1355.cc b/gdb/testsuite/gdb.cp/gdb1355.cc
new file mode 100644 (file)
index 0000000..a53ca20
--- /dev/null
@@ -0,0 +1,35 @@
+struct mystruct
+{
+  int m_int;
+  char m_char;
+  long int m_long_int;
+  unsigned int m_unsigned_int;
+  long unsigned int m_long_unsigned_int;
+  // long long int m_long_long_int;
+  // long long unsigned int m_long_long_unsigned_int;
+  short int m_short_int;
+  short unsigned int m_short_unsigned_int;
+  unsigned char m_unsigned_char;
+  float m_float;
+  double m_double;
+  long double m_long_double;
+  // complex int m_complex_int;
+  // complex float m_complex_float;
+  // complex long double m_complex_long_double;
+  // wchar_t m_wchar_t;
+  bool m_bool;
+};
+
+struct mystruct s1 =
+{
+    117, 'a', 118, 119, 120,
+    // 121, 122,
+    123, 124, 'b', 125.0, 126.0, 127.0,
+    // complex int, complex float, complex long double, wchar_t,
+    true
+};
+
+int main ()
+{
+  return 0;
+}
diff --git a/gdb/testsuite/gdb.cp/gdb1355.exp b/gdb/testsuite/gdb.cp/gdb1355.exp
new file mode 100644 (file)
index 0000000..11f16d5
--- /dev/null
@@ -0,0 +1,119 @@
+# Copyright 2003 Free Software Foundation, Inc.
+
+# 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.
+# 
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+# GNU General Public License for more details.
+# 
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.  
+
+# Tests for PR gdb/1355, which is a reference to PR gcc/12066.
+# 2003-08-26  Michael Chastain <mec@shout.net>
+
+# This file is part of the gdb testsuite.
+
+set ws "\[\r\n\t \]*"
+set nl "\[\r\n\]+"
+
+if $tracelevel then {
+    strace $tracelevel
+    }
+
+if { [skip_cplus_tests] } { continue }
+
+#
+# test running programs
+#
+set prms_id 0
+set bug_id 0
+
+set testfile "gdb1355"
+set srcfile ${testfile}.cc
+set binfile ${objdir}/${subdir}/${testfile}
+
+if  { [gdb_compile "${srcdir}/${subdir}/${srcfile}" "${binfile}" executable {debug c++}] != "" } {
+     gdb_suppress_entire_file "Testcase compile failed, so all tests in this file will automatically fail."
+}
+
+if [get_compiler_info ${binfile} "c++"] {
+    return -1
+}
+
+gdb_exit
+gdb_start
+gdb_reinitialize_dir $srcdir/$subdir
+gdb_load ${binfile}
+
+if ![runto_main] then {
+    perror "couldn't run to main"
+    continue
+} 
+
+# See http://sources.redhat.com/gdb/bugs/1355
+# See http://gcc.gnu.org/bugzilla/show_bug.cgi?id=12066
+#
+# g++ -gstabs+ does not emit stabs for fundamental types.
+# They get emitted later inside other types, so they have no names
+# and gdb cannot handle them.
+
+set s_head  "${ws}(struct|class) mystruct \{(${ws}public:|)"
+set s_tail  ".*"
+
+set f_i     "${ws}int m_int;"
+set f_c     "${ws}char m_char;"
+set f_li    "${ws}long int m_long_int;"
+set f_ui    "${ws}unsigned int m_unsigned_int;"
+set f_lui   "${ws}long unsigned int m_long_unsigned_int;"
+set f_si    "${ws}short int m_short_int;"
+set f_sui   "${ws}short unsigned int m_short_unsigned_int;"
+set f_uc    "${ws}unsigned char m_unsigned_char;"
+set f_f     "${ws}float m_float;"
+set f_d     "${ws}double m_double;"
+set f_ld    "${ws}long double m_long_double;"
+set f_b     "${ws}bool m_bool;"
+
+set itc     "<invalid type code ${decimal}>"
+set bad_i   "${ws}(${itc}|int) m_int;";
+set bad_c   "${ws}(${itc}|char) m_char;"
+set bad_li  "${ws}(${itc}|long int) m_long_int;"
+set bad_ui  "${ws}(${itc}|unsigned int) m_unsigned_int;"
+set bad_lui "${ws}(${itc}|long unsigned int) m_long_unsigned_int;"
+set bad_si  "${ws}(${itc}|short int) m_short_int;"
+set bad_sui "${ws}(${itc}|short unsigned int) m_short_unsigned_int;"
+set bad_uc  "${ws}(${itc}|unsigned char) m_unsigned_char;"
+set bad_f   "${ws}(${itc}|float) m_float;"
+set bad_d   "${ws}(${itc}|double) m_double;"
+set bad_ld  "${ws}(${itc}|long double) m_long_double;"
+set bad_b   "${ws}(${itc}|bool) m_bool;"
+
+gdb_test_multiple "ptype s1" "ptype s1" {
+    -re "type = ${s_head}${f_i}${f_c}${f_li}${f_ui}${f_lui}${f_si}${f_sui}${f_uc}${f_f}${f_d}${f_ld}${f_b}${s_tail}\}$nl$gdb_prompt $" {
+       pass "ptype s1"
+    }
+    -re "type = ${s_head}${bad_i}${bad_c}${bad_li}${bad_ui}${bad_lui}${bad_si}${bad_sui}${bad_uc}${bad_f}${bad_d}${bad_ld}${bad_b}${s_tail}\}$nl$gdb_prompt $" {
+       # This happened with gcc HEAD 2003-08-20 08:00:00 UTC, -gstabs+.
+       kfail "gdb/1355" "ptype s1"
+    }
+}
+
+gdb_test_multiple "print s1" "print s1" {
+    -re "$decimal = \{m_int = 117, m_char = 97 'a', m_long_int = 118, m_unsigned_int = 119, m_long_unsigned_int = 120, m_short_int = 123, m_short_unsigned_int = 124, m_unsigned_char = 98 'b', m_float = 125, m_double = 126, m_long_double = 127, m_bool = true\}$nl$gdb_prompt $" {
+       pass "print s1"
+    }
+    -re "$decimal = \{m_int = 117, m_char = 97 'a', m_long_int = 118, m_unsigned_int = 119, m_long_unsigned_int = 120, m_short_int = 123, m_short_unsigned_int = 124, m_unsigned_char = 98 'b', m_float = 125, m_double = 126, m_long_double = 127, m_bool = 117\}$nl$gdb_prompt $" {
+       # This pattern is very picky, but if more different output
+       # shows up, I can just add more arms.  -- chastain 2003-08-26
+       #
+       # This happened with gcc HEAD 2003-08-20 08:00:00 UTC, -gstabs+.
+       # Look at the value of m_bool.  It looks like gdb latched onto
+       # random int type and then used the data at structure offset 0.
+       kfail "gdb/1355" "print s1"
+    }
+}
diff --git a/gdb/testsuite/gdb.threads/switch-threads.c b/gdb/testsuite/gdb.threads/switch-threads.c
new file mode 100644 (file)
index 0000000..3e5a825
--- /dev/null
@@ -0,0 +1,47 @@
+/* A minimal multi-threaded test case.
+
+   Copyright 2003
+   Free Software Foundation, Inc.
+
+   This file is part of GDB.
+
+   This program is free software; you can redistribute it and/or modify
+   it under the terms of the GNU General Public License as published by
+   the Free Software Foundation; either version 2 of the License, or
+   (at your option) any later version.
+
+   This program is distributed in the hope that it will be useful,
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+   GNU General Public License for more details.
+
+   You should have received a copy of the GNU General Public License
+   along with this program; if not, write to the Free Software
+   Foundation, Inc., 59 Temple Place - Suite 330,
+   Boston, MA 02111-1307, USA.  */
+
+#include <pthread.h>
+
+void foo (void)
+{
+}
+
+void *thread_func (void *arg)
+{
+  int x;
+  for (x = 0; x < 10; x++)
+    foo ();
+  return 0;
+}
+
+int main()
+{
+  pthread_t thr;
+  void *ret;
+  int x;
+
+  pthread_create (&thr, NULL, thread_func, NULL);
+  pthread_join (thr, &ret);
+  for (x = 0; x < 10; x++)
+    foo ();  
+}
diff --git a/gdb/testsuite/gdb.threads/switch-threads.exp b/gdb/testsuite/gdb.threads/switch-threads.exp
new file mode 100644 (file)
index 0000000..d5608cf
--- /dev/null
@@ -0,0 +1,52 @@
+# Copyright (C) 2003 Free Software Foundation, Inc.
+
+# 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.
+# 
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+# GNU General Public License for more details.
+# 
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.  
+
+# Please email any bugs, comments, and/or additions to this file to:
+# bug-gdb@prep.ai.mit.edu
+
+# This file was written by Daniel Jacobowitz <drow@mvista.com>.
+#
+# It tests that the correct thread is single-stepped.  Prior to the
+# introduction of vCont, we didn't pass enough information to remote
+# multi-threaded stubs to reliably get this correct; gdbserver defaulted
+# to the first thread.
+
+# TODO: we should also test explicitly changing threads with the "thread"
+# command.
+
+if $tracelevel then {
+       strace $tracelevel
+}
+
+set testfile "switch-threads"
+set srcfile ${testfile}.c
+set binfile ${objdir}/${subdir}/${testfile}
+
+if {[gdb_compile_pthreads "${srcdir}/${subdir}/${srcfile}" "${binfile}" executable [list debug "incdir=${objdir}"]] != "" } {
+    return -1
+}
+
+gdb_exit
+gdb_start
+gdb_reinitialize_dir $srcdir/$subdir
+gdb_load ${binfile}
+
+runto_main
+
+gdb_breakpoint thread_func
+gdb_continue_to_breakpoint "continue to thread_func"
+gdb_test "next" ".*foo \\(\\);"
+
diff --git a/include/gdb/sim-frv.h b/include/gdb/sim-frv.h
new file mode 100644 (file)
index 0000000..0a1e021
--- /dev/null
@@ -0,0 +1,53 @@
+/* This file defines the interface between the FR-V simulator and GDB.
+
+   Copyright 2003 Free Software Foundation, Inc.
+
+   Contributed by Red Hat.
+
+   This file is part of GDB.
+
+   This program is free software; you can redistribute it and/or
+   modify it under the terms of the GNU General Public License as
+   published by the Free Software Foundation; either version 2 of the
+   License, or (at your option) any later version.
+
+   This program is distributed in the hope that it will be useful, but
+   WITHOUT ANY WARRANTY; without even the implied warranty of
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+   General Public License for more details.
+
+   You should have received a copy of the GNU General Public License
+   along with this program; if not, write to the Free Software
+   Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
+   02111-1307, USA.  */
+
+#if !defined (SIM_FRV_H)
+#define SIM_FRV_H
+
+#ifdef __cplusplus
+extern "C" { // }
+#endif
+
+enum sim_frv_regs
+{
+  SIM_FRV_GR0_REGNUM  = 0,
+  SIM_FRV_GR63_REGNUM = 63,
+  SIM_FRV_FR0_REGNUM  = 64,
+  SIM_FRV_FR63_REGNUM = 127,
+  SIM_FRV_PC_REGNUM   = 128,
+
+  /* An FR-V architecture may have up to 4096 special purpose registers
+     (SPRs).  In order to determine a specific constant used to access
+     a particular SPR, one of the H_SPR_ prefixed offsets defined in
+     opcodes/frv-desc.h should be added to SIM_FRV_SPR0_REGNUM.  So,
+     for example, the number that GDB uses to fetch the link register
+     from the simulator is (SIM_FRV_SPR0_REGNUM + H_SPR_LR).  */
+  SIM_FRV_SPR0_REGNUM = 129,
+  SIM_FRV_SPR4095_REGNUM = SIM_FRV_SPR0_REGNUM + 4095
+};
+
+#ifdef __cplusplus
+}
+#endif
+
+#endif