]> git.ipfire.org Git - thirdparty/binutils-gdb.git/blobdiff - sim/common/run.c
Report SIGBUS and halt simulation when ld/st detect a misaligned address.
[thirdparty/binutils-gdb.git] / sim / common / run.c
index 982d99c7ad88a734c225181d4695be3819332a20..1a4e96908f11686d361c6888374454d4a714fcc4 100644 (file)
 /* run front end support for all the simulators.
-   Copyright (C) 1992, 1993 1994, 1995 Free Software Foundation, Inc.
+   Copyright (C) 1992, 93-96, 1997 Free Software Foundation, Inc.
 
-GNU CC is free software; you can redistribute it and/or modify
+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, or (at your option)
 any later version.
 
-GNU CC is distributed in the hope that it will be useful,
+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.  */
+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.  */
 
+/* Steve Chamberlain sac@cygnus.com,
+   and others at Cygnus.  */
 
-/* Steve Chamberlain
-   sac@cygnus.com */
+#include "config.h"
+#include "tconfig.h"
 
 #include <signal.h>
 #include <stdio.h>
+#ifdef __STDC__
+#include <stdarg.h>
+#else
 #include <varargs.h>
+#endif
+
+#ifdef HAVE_STDLIB_H
+#include <stdlib.h>
+#endif
+
+#ifdef HAVE_STRING_H
+#include <string.h>
+#else
+#ifdef HAVE_STRINGS_H
+#include <strings.h>
+#endif
+#endif
+
+#include "libiberty.h"
 #include "bfd.h"
-#include "remote-sim.h"
 #include "callback.h"
-#ifndef SIGQUIT
-#define SIGQUIT SIGTERM
-#endif
+#include "remote-sim.h"
+#include "ansidecl.h"
+
+#include "../libiberty/alloca-conf.h"
 
-void usage();
+static void usage PARAMS ((void));
 extern int optind;
 extern char *optarg;
 
-bfd *sim_bfd;
+extern host_callback default_callback;
 
-int target_byte_order;
+static char *myname;
+
+/* NOTE: sim_size() and sim_trace() are going away */
+extern int sim_trace PARAMS ((SIM_DESC sd));
+
+extern int getopt ();
+
+#ifdef NEED_UI_LOOP_HOOK
+/* Gdb foolery. This is only needed for gdb using a gui. */
+int (*ui_loop_hook) PARAMS ((int signo));
+#endif
+
+static SIM_DESC sd;
+
+static RETSIGTYPE
+cntrl_c (int sig ATTRIBUTE_UNUSED)
+{
+  if (! sim_stop (sd))
+    {
+      fprintf (stderr, "Quit!\n");
+      exit (1);
+    }
+}
 
-extern host_callback default_callback;
 int
 main (ac, av)
      int ac;
      char **av;
 {
+  RETSIGTYPE (*prev_sigint) ();
   bfd *abfd;
-  bfd_vma start_address;
-  asection *s;
   int i;
   int verbose = 0;
   int trace = 0;
-  char *name = "";
+  char *name;
+  static char *no_args[4];
+  char **sim_argv = &no_args[0];
+  char **prog_args;
   enum sim_stop reason;
   int sigrc;
 
-  while ((i = getopt (ac, av, "m:p:s:tv")) != EOF) 
+  myname = av[0] + strlen (av[0]);
+  while (myname > av[0] && myname[-1] != '/')
+    --myname;
+
+  /* The first element of sim_open's argv is the program name.  */
+  no_args[0] = av[0];
+#ifdef SIM_HAVE_BIENDIAN
+  no_args[1] = "-E";
+  no_args[2] = "set-later";
+#endif
+
+  /* FIXME: This is currently being migrated into sim_open.
+     Simulators that use functions such as sim_size() still require
+     this. */
+  default_callback.init (&default_callback);
+  sim_set_callbacks (&default_callback);
+
+  /* FIXME: This is currently being rewritten to have each simulator
+     do all argv processing.  */
+
+#ifdef SIM_H8300 /* FIXME: quick hack */
+  while ((i = getopt (ac, av, "a:c:m:p:s:htv")) != EOF) 
+#else
+  while ((i = getopt (ac, av, "a:c:m:p:s:tv")) != EOF) 
+#endif
     switch (i)
       {
+      case 'a':
+       /* FIXME: Temporary hack.  */
+       {
+         int len = strlen (av[0]) + strlen (optarg);
+         char *argbuf = (char *) alloca (len + 2 + 50);
+         sprintf (argbuf, "%s %s", av[0], optarg);
+#ifdef SIM_HAVE_BIENDIAN
+         /* The desired endianness must be passed to sim_open.
+            The value for "set-later" is set when we know what it is.
+            -E support isn't yet part of the published interface.  */
+         strcat (argbuf, " -E set-later");
+#endif
+         sim_argv = buildargv (argbuf);
+       }
+       break;
+#ifdef SIM_HAVE_SIMCACHE
+      case 'c':
+       sim_set_simcache_size (atoi (optarg));
+       break;
+#endif
       case 'm':
+       /* FIXME: Rename to sim_set_mem_size.  */
        sim_size (atoi (optarg));
        break;
+#ifdef SIM_HAVE_PROFILE
       case 'p':
        sim_set_profile (atoi (optarg));
        break;
       case 's':
        sim_set_profile_size (atoi (optarg));
        break;
+#endif
       case 't':
        trace = 1;
+       /* FIXME: need to allow specification of what to trace.  */
+       /* sim_set_trace (1); */
        break;
       case 'v':
+       /* Things that are printed with -v are the kinds of things that
+          gcc -v prints.  This is not meant to include detailed tracing
+          or debugging information, just summaries.  */
        verbose = 1;
+       /* sim_set_verbose (1); */
        break;
+       /* FIXME: Quick hack, to be replaced by more general facility.  */
+#ifdef SIM_H8300
+      case 'h':
+       set_h8300h (1);
+       break;
+#endif
       default:
-       usage();
+       usage ();
       }
+
   ac -= optind;
   av += optind;
-
-  if (ac != 1)
-    usage();
+  if (ac <= 0)
+    usage ();
 
   name = *av;
+  prog_args = av;
 
   if (verbose)
     {
-      printf ("run %s\n", name);
+      printf ("%s %s\n", myname, name);
     }
 
-  sim_bfd = abfd = bfd_openr (name, 0);
+  abfd = bfd_openr (name, 0);
   if (!abfd) 
     {
-      fprintf (stderr, "run: can't open %s: %s\n", 
-             name, bfd_errmsg(bfd_get_error()));
+      fprintf (stderr, "%s: can't open %s: %s\n", 
+              myname, name, bfd_errmsg (bfd_get_error ()));
       exit (1);
     }
 
   if (!bfd_check_format (abfd, bfd_object))
     {
-      fprintf (stderr, "run: can't load %s: %s\n",
-              name, bfd_errmsg(bfd_get_error()));
+      fprintf (stderr, "%s: can't load %s: %s\n",
+              myname, name, bfd_errmsg (bfd_get_error ()));
       exit (1);
     }
 
-  sim_set_callbacks (&default_callback);
-  default_callback.init (&default_callback);
+#ifdef SIM_HAVE_BIENDIAN
+  /* The endianness must be passed to sim_open because one may wish to
+     examine/set registers before calling sim_load [which is the other
+     place where one can determine endianness].  We previously passed the
+     endianness via global `target_byte_order' but that's not a clean
+     interface.  */
+  for (i = 1; sim_argv[i + 1] != NULL; ++i)
+    continue;
+  if (bfd_big_endian (abfd))
+    sim_argv[i] = "big";
+  else
+    sim_argv[i] = "little";
+#endif
 
   /* Ensure that any run-time initialisation that needs to be
      performed by the simulator can occur. */
-  sim_open(NULL);
+  sd = sim_open (SIM_OPEN_STANDALONE, &default_callback, abfd, sim_argv);
+  if (sd == 0)
+    exit (1);
 
-  for (s = abfd->sections; s; s = s->next)
-  if (abfd && (s->flags & SEC_LOAD))
-    {
-      unsigned char *buffer = (unsigned char *)malloc (bfd_section_size (abfd, s));
-      bfd_get_section_contents (abfd,
-                               s,
-                               buffer,
-                               0,
-                               bfd_section_size (abfd, s));
-      sim_write (s->vma, buffer, bfd_section_size (abfd, s));
-    }
+  if (sim_load (sd, name, abfd, 0) == SIM_RC_FAIL)
+    exit (1);
 
-  start_address = bfd_get_start_address (abfd);
-  sim_create_inferior (start_address, NULL, NULL);
-
-  target_byte_order = bfd_big_endian (abfd) ? 4321 : 1234;
+  if (sim_create_inferior (sd, abfd, prog_args, NULL) == SIM_RC_FAIL)
+    exit (1);
 
   if (trace)
     {
       int done = 0;
+      prev_sigint = signal (SIGINT, cntrl_c);
       while (!done)
        {
-         done = sim_trace ();
+         done = sim_trace (sd);
        }
+      signal (SIGINT, prev_sigint);
     }
   else
     {
-      sim_resume (0, 0);
+      do
+       {
+         prev_sigint = signal (SIGINT, cntrl_c);
+         sim_resume (sd, 0, sigrc);
+         signal (SIGINT, prev_sigint);
+         sim_stop_reason (sd, &reason, &sigrc);
+       }
+      while (reason == sim_stopped && sigrc != SIGINT);
     }
-  if (verbose)
-    sim_info (0);
-
-  sim_stop_reason (&reason, &sigrc);
 
-  sim_close(0);
+  if (verbose)
+    sim_info (sd, 0);
+  sim_close (sd, 0);
 
   /* If reason is sim_exited, then sigrc holds the exit code which we want
      to return.  If reason is sim_stopped or sim_signalled, then sigrc holds
      the signal that the simulator received; we want to return that to
      indicate failure.  */
+
+#ifdef SIM_H8300 /* FIXME: Ugh.  grep for SLEEP in compile.c  */
+  if (sigrc == SIGILL)
+    abort ();
+  sigrc = 0;
+#else
+  /* Why did we stop? */
+  switch (reason)
+    {
+    case sim_signalled:
+    case sim_stopped:
+      if (sigrc != 0)
+        fprintf (stderr, "program stopped with signal %d.\n", sigrc);
+      break;
+
+    case sim_exited:
+      break;
+
+    case sim_running:
+    case sim_polling: /* these indicate a serious problem */
+      abort ();
+      break;
+
+    }
+#endif
+
   return sigrc;
 }
 
-void
-usage()
+static void
+usage ()
 {
-  fprintf (stderr, "usage: run [-tv][-m size] program\n");
+  fprintf (stderr, "Usage: %s [options] program [program args]\n", myname);
+  fprintf (stderr, "Options:\n");
+  fprintf (stderr, "-a args         Pass `args' to simulator.\n");
+#ifdef SIM_HAVE_SIMCACHE
+  fprintf (stderr, "-c size         Set simulator cache size to `size'.\n");
+#endif
+#ifdef SIM_H8300
+  fprintf (stderr, "-h              Executable is for h8/300h or h8/300s.\n");
+#endif
+  fprintf (stderr, "-m size         Set memory size of simulator, in bytes.\n");
+#ifdef SIM_HAVE_PROFILE
+  fprintf (stderr, "-p freq         Set profiling frequency.\n");
+  fprintf (stderr, "-s size         Set profiling size.\n");
+#endif
+  fprintf (stderr, "-t              Perform instruction tracing.\n");
+  fprintf (stderr, "                Note: Very few simulators support tracing.\n");
+  fprintf (stderr, "-v              Verbose output.\n");
+  fprintf (stderr, "\n");
+  fprintf (stderr, "program args    Arguments to pass to simulated program.\n");
+  fprintf (stderr, "                Note: Very few simulators support this.\n");
   exit (1);
 }
-
-