123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332 |
- /* This file is part of the program psim.
- Copyright (C) 1994-1997, Andrew Cagney <cagney@highland.com.au>
- This program is free software; you can redistribute it and/or modify
- it under the terms of the GNU General Public License as published by
- the Free Software Foundation; either version 3 of the License, or
- (at your option) any later version.
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU General Public License for more details.
-
- You should have received a copy of the GNU General Public License
- along with this program; if not, see <http://www.gnu.org/licenses/>.
-
- */
- #include <stdarg.h>
- #include <stdio.h>
- #include <fcntl.h>
- #include <signal.h>
- #include "psim.h"
- #include "options.h"
- #include "device.h" /* FIXME: psim should provide the interface */
- #include "events.h" /* FIXME: psim should provide the interface */
- #include "bfd.h"
- #include "gdb/callback.h"
- #include "gdb/remote-sim.h"
- #ifdef HAVE_STDLIB_H
- #include <stdlib.h>
- #endif
- #ifdef HAVE_UNISTD_H
- #include <unistd.h>
- #endif
- #ifdef HAVE_STRING_H
- #include <string.h>
- #else
- #ifdef HAVE_STRINGS_H
- #include <strings.h>
- #endif
- #endif
- #include <errno.h>
- #if !defined(O_NDELAY) || !defined(F_GETFL) || !defined(F_SETFL)
- #undef WITH_STDIO
- #define WITH_STDIO DO_USE_STDIO
- #endif
- extern char **environ;
- static psim *simulation = NULL;
- void
- sim_io_poll_quit (void)
- {
- /* nothing to do */
- }
- void
- sim_io_printf_filtered(const char *msg, ...)
- {
- va_list ap;
- va_start(ap, msg);
- vprintf(msg, ap);
- va_end(ap);
- }
- void
- error (const char *msg, ...)
- {
- va_list ap;
- va_start(ap, msg);
- vprintf(msg, ap);
- printf("\n");
- va_end(ap);
- /* any final clean up */
- if (ppc_trace[trace_print_info] && simulation != NULL)
- psim_print_info (simulation, ppc_trace[trace_print_info]);
- exit (1);
- }
- int
- sim_io_write_stdout(const char *buf,
- int sizeof_buf)
- {
- switch (CURRENT_STDIO) {
- case DO_USE_STDIO:
- {
- int i;
- for (i = 0; i < sizeof_buf; i++) {
- putchar(buf[i]);
- }
- return i;
- }
- break;
- case DONT_USE_STDIO:
- return write(1, buf, sizeof_buf);
- break;
- default:
- error("sim_io_write_stdout: invalid switch\n");
- }
- return 0;
- }
- int
- sim_io_write_stderr(const char *buf,
- int sizeof_buf)
- {
- switch (CURRENT_STDIO) {
- case DO_USE_STDIO:
- {
- int i;
- for (i = 0; i < sizeof_buf; i++) {
- fputc(buf[i], stderr);
- }
- return i;
- }
- break;
- case DONT_USE_STDIO:
- return write(2, buf, sizeof_buf);
- break;
- default:
- error("sim_io_write_stdout: invalid switch\n");
- }
- return 0;
- }
- int
- sim_io_read_stdin(char *buf,
- int sizeof_buf)
- {
- switch (CURRENT_STDIO) {
- case DO_USE_STDIO:
- if (sizeof_buf > 1) {
- if (fgets(buf, sizeof_buf, stdin) != NULL)
- return strlen(buf);
- }
- else if (sizeof_buf == 1) {
- char b[2];
- if (fgets(b, sizeof(b), stdin) != NULL) {
- memcpy(buf, b, strlen(b));
- return strlen(b);
- }
- }
- else if (sizeof_buf == 0)
- return 0;
- return sim_io_eof;
- break;
- case DONT_USE_STDIO:
- #if defined(O_NDELAY) && defined(F_GETFL) && defined(F_SETFL)
- {
- /* check for input */
- int flags;
- int status;
- int nr_read;
- int result;
- /* get the old status */
- flags = fcntl(0, F_GETFL, 0);
- if (flags == -1) {
- perror("sim_io_read_stdin");
- return sim_io_eof;
- }
- /* temp, disable blocking IO */
- status = fcntl(0, F_SETFL, flags | O_NDELAY);
- if (status == -1) {
- perror("sim_io_read_stdin");
- return sim_io_eof;
- }
- /* try for input */
- nr_read = read(0, buf, sizeof_buf);
- if (nr_read > 0
- || (nr_read == 0 && sizeof_buf == 0))
- result = nr_read;
- else if (nr_read == 0)
- result = sim_io_eof;
- else { /* nr_read < 0 */
- if (errno == EAGAIN)
- result = sim_io_not_ready;
- else
- result = sim_io_eof;
- }
- /* return to regular vewing */
- status = fcntl(0, F_SETFL, flags);
- if (status == -1) {
- perror("sim_io_read_stdin");
- return sim_io_eof;
- }
- return result;
- }
- break;
- #endif
- default:
- error("sim_io_read_stdin: invalid switch\n");
- break;
- }
- return 0;
- }
- void
- sim_io_flush_stdoutput(void)
- {
- switch (CURRENT_STDIO) {
- case DO_USE_STDIO:
- fflush (stdout);
- break;
- case DONT_USE_STDIO:
- break;
- default:
- error("sim_io_flush_stdoutput: invalid switch\n");
- break;
- }
- }
- void
- sim_io_error (SIM_DESC sd, const char *msg, ...)
- {
- va_list ap;
- va_start(ap, msg);
- vprintf(msg, ap);
- printf("\n");
- va_end(ap);
- /* any final clean up */
- if (ppc_trace[trace_print_info] && simulation != NULL)
- psim_print_info (simulation, ppc_trace[trace_print_info]);
- exit (1);
- }
- void *
- zalloc(long size)
- {
- void *memory = malloc(size);
- if (memory == NULL)
- error("zalloc failed\n");
- memset(memory, 0, size);
- return memory;
- }
- /* When a CNTRL-C occures, queue an event to shut down the simulation */
- static RETSIGTYPE
- cntrl_c(int sig)
- {
- psim_stop (simulation);
- }
- int
- main(int argc, char **argv)
- {
- const char *name_of_file;
- char *arg_;
- psim_status status;
- device *root = psim_tree();
- /* parse the arguments */
- argv = psim_options(root, argv + 1);
- if (argv[0] == NULL) {
- if (ppc_trace[trace_opts]) {
- print_options ();
- return 0;
- } else {
- psim_usage(0, 0);
- }
- }
- name_of_file = argv[0];
- if (ppc_trace[trace_opts])
- print_options ();
- /* create the simulator */
- simulation = psim_create(name_of_file, root);
- /* fudge the environment so that _=prog-name */
- arg_ = (char*)zalloc(strlen(argv[0]) + strlen("_=") + 1);
- strcpy(arg_, "_=");
- strcat(arg_, argv[0]);
- putenv(arg_);
- /* initialize it */
- psim_init(simulation);
- psim_stack(simulation, argv, environ);
- {
- RETSIGTYPE (*prev) ();
- prev = signal(SIGINT, cntrl_c);
- psim_run(simulation);
- signal(SIGINT, prev);
- }
- /* any final clean up */
- if (ppc_trace[trace_print_info])
- psim_print_info (simulation, ppc_trace[trace_print_info]);
- /* why did we stop */
- status = psim_get_status(simulation);
- switch (status.reason) {
- case was_continuing:
- error("psim: continuing while stopped!\n");
- return 0;
- case was_trap:
- error("psim: no trap insn\n");
- return 0;
- case was_exited:
- return status.signal;
- case was_signalled:
- printf ("%s: Caught signal %d at address 0x%lx\n",
- name_of_file, (int)status.signal,
- (long)status.program_counter);
- return status.signal;
- default:
- error("unknown halt condition\n");
- return 0;
- }
- }
|