1 /* Simulator option handling.
2    Copyright (C) 1996-2020 Free Software Foundation, Inc.
3    Contributed by Cygnus Support.
4 
5 This file is part of GDB, the GNU debugger.
6 
7 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 3 of the License, or
10 (at your option) any later version.
11 
12 This program is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15 GNU General Public License for more details.
16 
17 You should have received a copy of the GNU General Public License
18 along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
19 
20 #include "config.h"
21 #include "sim-main.h"
22 #ifdef HAVE_STRING_H
23 #include <string.h>
24 #else
25 #ifdef HAVE_STRINGS_H
26 #include <strings.h>
27 #endif
28 #endif
29 #ifdef HAVE_STDLIB_H
30 #include <stdlib.h>
31 #endif
32 #include <ctype.h>
33 #include <stdio.h>
34 #include "libiberty.h"
35 #include "sim-options.h"
36 #include "sim-io.h"
37 #include "sim-assert.h"
38 #include "version.h"
39 
40 #include "bfd.h"
41 
42 /* Add a set of options to the simulator.
43    TABLE is an array of OPTIONS terminated by a NULL `opt.name' entry.
44    This is intended to be called by modules in their `install' handler.  */
45 
46 SIM_RC
sim_add_option_table(SIM_DESC sd,sim_cpu * cpu,const OPTION * table)47 sim_add_option_table (SIM_DESC sd, sim_cpu *cpu, const OPTION *table)
48 {
49   struct option_list *ol = ((struct option_list *)
50 			    xmalloc (sizeof (struct option_list)));
51 
52   /* Note: The list is constructed in the reverse order we're called so
53      later calls will override earlier ones (in case that ever happens).
54      This is the intended behaviour.  */
55 
56   if (cpu)
57     {
58       ol->next = CPU_OPTIONS (cpu);
59       ol->options = table;
60       CPU_OPTIONS (cpu) = ol;
61     }
62   else
63     {
64       ol->next = STATE_OPTIONS (sd);
65       ol->options = table;
66       STATE_OPTIONS (sd) = ol;
67     }
68 
69   return SIM_RC_OK;
70 }
71 
72 /* Standard option table.
73    Modules may specify additional ones.
74    The caller of sim_parse_args may also specify additional options
75    by calling sim_add_option_table first.  */
76 
77 static DECLARE_OPTION_HANDLER (standard_option_handler);
78 
79 /* FIXME: We shouldn't print in --help output options that aren't usable.
80    Some fine tuning will be necessary.  One can either move less general
81    options to another table or use a HAVE_FOO macro to ifdef out unavailable
82    options.  */
83 
84 /* ??? One might want to conditionally compile out the entries that
85    aren't enabled.  There's a distinction, however, between options a
86    simulator can't support and options that haven't been configured in.
87    Certainly options a simulator can't support shouldn't appear in the
88    output of --help.  Whether the same thing applies to options that haven't
89    been configured in or not isn't something I can get worked up over.
90    [Note that conditionally compiling them out might simply involve moving
91    the option to another table.]
92    If you decide to conditionally compile them out as well, delete this
93    comment and add a comment saying that that is the rule.  */
94 
95 typedef enum {
96   OPTION_DEBUG_INSN = OPTION_START,
97   OPTION_DEBUG_FILE,
98   OPTION_DO_COMMAND,
99   OPTION_ARCHITECTURE,
100   OPTION_TARGET,
101   OPTION_ARCHITECTURE_INFO,
102   OPTION_ENVIRONMENT,
103   OPTION_ALIGNMENT,
104   OPTION_VERBOSE,
105   OPTION_ENDIAN,
106   OPTION_DEBUG,
107   OPTION_HELP,
108   OPTION_VERSION,
109   OPTION_LOAD_LMA,
110   OPTION_LOAD_VMA,
111   OPTION_SYSROOT
112 } STANDARD_OPTIONS;
113 
114 static const OPTION standard_options[] =
115 {
116   { {"verbose", no_argument, NULL, OPTION_VERBOSE},
117       'v', NULL, "Verbose output",
118       standard_option_handler, NULL },
119 
120   { {"endian", required_argument, NULL, OPTION_ENDIAN},
121       'E', "big|little", "Set endianness",
122       standard_option_handler, NULL },
123 
124   /* This option isn't supported unless all choices are supported in keeping
125      with the goal of not printing in --help output things the simulator can't
126      do [as opposed to things that just haven't been configured in].  */
127   { {"environment", required_argument, NULL, OPTION_ENVIRONMENT},
128       '\0', "user|virtual|operating", "Set running environment",
129       standard_option_handler },
130 
131   { {"alignment", required_argument, NULL, OPTION_ALIGNMENT},
132       '\0', "strict|nonstrict|forced", "Set memory access alignment",
133       standard_option_handler },
134 
135   { {"debug", no_argument, NULL, OPTION_DEBUG},
136       'D', NULL, "Print debugging messages",
137       standard_option_handler },
138   { {"debug-insn", no_argument, NULL, OPTION_DEBUG_INSN},
139       '\0', NULL, "Print instruction debugging messages",
140       standard_option_handler },
141   { {"debug-file", required_argument, NULL, OPTION_DEBUG_FILE},
142       '\0', "FILE NAME", "Specify debugging output file",
143       standard_option_handler },
144 
145   { {"do-command", required_argument, NULL, OPTION_DO_COMMAND},
146       '\0', "COMMAND", ""/*undocumented*/,
147       standard_option_handler },
148 
149   { {"help", no_argument, NULL, OPTION_HELP},
150       'H', NULL, "Print help information",
151       standard_option_handler },
152   { {"version", no_argument, NULL, OPTION_VERSION},
153       '\0', NULL, "Print version information",
154       standard_option_handler },
155 
156   { {"architecture", required_argument, NULL, OPTION_ARCHITECTURE},
157       '\0', "MACHINE", "Specify the architecture to use",
158       standard_option_handler },
159   { {"architecture-info", no_argument, NULL, OPTION_ARCHITECTURE_INFO},
160       '\0', NULL, "List supported architectures",
161       standard_option_handler },
162   { {"info-architecture", no_argument, NULL, OPTION_ARCHITECTURE_INFO},
163       '\0', NULL, NULL,
164       standard_option_handler },
165 
166   { {"target", required_argument, NULL, OPTION_TARGET},
167       '\0', "BFDNAME", "Specify the object-code format for the object files",
168       standard_option_handler },
169 
170   { {"load-lma", no_argument, NULL, OPTION_LOAD_LMA},
171       '\0', NULL,
172     "Use VMA or LMA addresses when loading image (default LMA)",
173       standard_option_handler, "load-{lma,vma}" },
174   { {"load-vma", no_argument, NULL, OPTION_LOAD_VMA},
175       '\0', NULL, "", standard_option_handler,  "" },
176 
177   { {"sysroot", required_argument, NULL, OPTION_SYSROOT},
178       '\0', "SYSROOT",
179     "Root for system calls with absolute file-names and cwd at start",
180       standard_option_handler, NULL },
181 
182   { {NULL, no_argument, NULL, 0}, '\0', NULL, NULL, NULL, NULL }
183 };
184 
185 static SIM_RC
standard_option_handler(SIM_DESC sd,sim_cpu * cpu,int opt,char * arg,int is_command)186 standard_option_handler (SIM_DESC sd, sim_cpu *cpu, int opt,
187 			 char *arg, int is_command)
188 {
189   int i,n;
190 
191   switch ((STANDARD_OPTIONS) opt)
192     {
193     case OPTION_VERBOSE:
194       STATE_VERBOSE_P (sd) = 1;
195       break;
196 
197     case OPTION_ENDIAN:
198       if (strcmp (arg, "big") == 0)
199 	{
200 	  if (WITH_TARGET_BYTE_ORDER == BFD_ENDIAN_LITTLE)
201 	    {
202 	      sim_io_eprintf (sd, "Simulator compiled for little endian only.\n");
203 	      return SIM_RC_FAIL;
204 	    }
205 	  /* FIXME:wip: Need to set something in STATE_CONFIG.  */
206 	  current_target_byte_order = BFD_ENDIAN_BIG;
207 	}
208       else if (strcmp (arg, "little") == 0)
209 	{
210 	  if (WITH_TARGET_BYTE_ORDER == BFD_ENDIAN_BIG)
211 	    {
212 	      sim_io_eprintf (sd, "Simulator compiled for big endian only.\n");
213 	      return SIM_RC_FAIL;
214 	    }
215 	  /* FIXME:wip: Need to set something in STATE_CONFIG.  */
216 	  current_target_byte_order = BFD_ENDIAN_LITTLE;
217 	}
218       else
219 	{
220 	  sim_io_eprintf (sd, "Invalid endian specification `%s'\n", arg);
221 	  return SIM_RC_FAIL;
222 	}
223       break;
224 
225     case OPTION_ENVIRONMENT:
226       if (strcmp (arg, "user") == 0)
227 	STATE_ENVIRONMENT (sd) = USER_ENVIRONMENT;
228       else if (strcmp (arg, "virtual") == 0)
229 	STATE_ENVIRONMENT (sd) = VIRTUAL_ENVIRONMENT;
230       else if (strcmp (arg, "operating") == 0)
231 	STATE_ENVIRONMENT (sd) = OPERATING_ENVIRONMENT;
232       else
233 	{
234 	  sim_io_eprintf (sd, "Invalid environment specification `%s'\n", arg);
235 	  return SIM_RC_FAIL;
236 	}
237       if (WITH_ENVIRONMENT != ALL_ENVIRONMENT
238 	  && WITH_ENVIRONMENT != STATE_ENVIRONMENT (sd))
239 	{
240 	  const char *type;
241 	  switch (WITH_ENVIRONMENT)
242 	    {
243 	    case USER_ENVIRONMENT: type = "user"; break;
244 	    case VIRTUAL_ENVIRONMENT: type = "virtual"; break;
245 	    case OPERATING_ENVIRONMENT: type = "operating"; break;
246 	    }
247 	  sim_io_eprintf (sd, "Simulator compiled for the %s environment only.\n",
248 			  type);
249 	  return SIM_RC_FAIL;
250 	}
251       break;
252 
253     case OPTION_ALIGNMENT:
254       if (strcmp (arg, "strict") == 0)
255 	{
256 	  if (WITH_ALIGNMENT == 0 || WITH_ALIGNMENT == STRICT_ALIGNMENT)
257 	    {
258 	      current_alignment = STRICT_ALIGNMENT;
259 	      break;
260 	    }
261 	}
262       else if (strcmp (arg, "nonstrict") == 0)
263 	{
264 	  if (WITH_ALIGNMENT == 0 || WITH_ALIGNMENT == NONSTRICT_ALIGNMENT)
265 	    {
266 	      current_alignment = NONSTRICT_ALIGNMENT;
267 	      break;
268 	    }
269 	}
270       else if (strcmp (arg, "forced") == 0)
271 	{
272 	  if (WITH_ALIGNMENT == 0 || WITH_ALIGNMENT == FORCED_ALIGNMENT)
273 	    {
274 	      current_alignment = FORCED_ALIGNMENT;
275 	      break;
276 	    }
277 	}
278       else
279 	{
280 	  sim_io_eprintf (sd, "Invalid alignment specification `%s'\n", arg);
281 	  return SIM_RC_FAIL;
282 	}
283       switch (WITH_ALIGNMENT)
284 	{
285 	case STRICT_ALIGNMENT:
286 	  sim_io_eprintf (sd, "Simulator compiled for strict alignment only.\n");
287 	  break;
288 	case NONSTRICT_ALIGNMENT:
289 	  sim_io_eprintf (sd, "Simulator compiled for nonstrict alignment only.\n");
290 	  break;
291 	case FORCED_ALIGNMENT:
292 	  sim_io_eprintf (sd, "Simulator compiled for forced alignment only.\n");
293 	  break;
294 	}
295       return SIM_RC_FAIL;
296 
297     case OPTION_DEBUG:
298       if (! WITH_DEBUG)
299 	sim_io_eprintf (sd, "Debugging not compiled in, `-D' ignored\n");
300       else
301 	{
302 	  for (n = 0; n < MAX_NR_PROCESSORS; ++n)
303 	    for (i = 0; i < MAX_DEBUG_VALUES; ++i)
304 	      CPU_DEBUG_FLAGS (STATE_CPU (sd, n))[i] = 1;
305 	}
306       break;
307 
308     case OPTION_DEBUG_INSN :
309       if (! WITH_DEBUG)
310 	sim_io_eprintf (sd, "Debugging not compiled in, `--debug-insn' ignored\n");
311       else
312 	{
313 	  for (n = 0; n < MAX_NR_PROCESSORS; ++n)
314 	    CPU_DEBUG_FLAGS (STATE_CPU (sd, n))[DEBUG_INSN_IDX] = 1;
315 	}
316       break;
317 
318     case OPTION_DEBUG_FILE :
319       if (! WITH_DEBUG)
320 	sim_io_eprintf (sd, "Debugging not compiled in, `--debug-file' ignored\n");
321       else
322 	{
323 	  FILE *f = fopen (arg, "w");
324 
325 	  if (f == NULL)
326 	    {
327 	      sim_io_eprintf (sd, "Unable to open debug output file `%s'\n", arg);
328 	      return SIM_RC_FAIL;
329 	    }
330 	  for (n = 0; n < MAX_NR_PROCESSORS; ++n)
331 	    CPU_DEBUG_FILE (STATE_CPU (sd, n)) = f;
332 	}
333       break;
334 
335     case OPTION_DO_COMMAND:
336       sim_do_command (sd, arg);
337       break;
338 
339     case OPTION_ARCHITECTURE:
340       {
341 	const struct bfd_arch_info *ap = bfd_scan_arch (arg);
342 	if (ap == NULL)
343 	  {
344 	    sim_io_eprintf (sd, "Architecture `%s' unknown\n", arg);
345 	    return SIM_RC_FAIL;
346 	  }
347 	STATE_ARCHITECTURE (sd) = ap;
348 	break;
349       }
350 
351     case OPTION_ARCHITECTURE_INFO:
352       {
353 	const char **list = bfd_arch_list ();
354 	const char **lp;
355 	if (list == NULL)
356 	  abort ();
357 	sim_io_printf (sd, "Possible architectures:");
358 	for (lp = list; *lp != NULL; lp++)
359 	  sim_io_printf (sd, " %s", *lp);
360 	sim_io_printf (sd, "\n");
361 	free (list);
362 	break;
363       }
364 
365     case OPTION_TARGET:
366       {
367 	STATE_TARGET (sd) = xstrdup (arg);
368 	break;
369       }
370 
371     case OPTION_LOAD_LMA:
372       {
373 	STATE_LOAD_AT_LMA_P (sd) = 1;
374 	break;
375       }
376 
377     case OPTION_LOAD_VMA:
378       {
379 	STATE_LOAD_AT_LMA_P (sd) = 0;
380 	break;
381       }
382 
383     case OPTION_HELP:
384       sim_print_help (sd, is_command);
385       if (STATE_OPEN_KIND (sd) == SIM_OPEN_STANDALONE)
386 	exit (0);
387       /* FIXME: 'twould be nice to do something similar if gdb.  */
388       break;
389 
390     case OPTION_VERSION:
391       sim_io_printf (sd, "GNU simulator %s%s\n", PKGVERSION, version);
392       if (STATE_OPEN_KIND (sd) == SIM_OPEN_STANDALONE)
393 	exit (0);
394       break;
395 
396     case OPTION_SYSROOT:
397       /* Don't leak memory in the odd event that there's lots of
398 	 --sysroot=... options.  We treat "" specially since this
399 	 is the statically initialized value and cannot free it.  */
400       if (simulator_sysroot[0] != '\0')
401 	free (simulator_sysroot);
402       if (arg[0] != '\0')
403 	simulator_sysroot = xstrdup (arg);
404       else
405 	simulator_sysroot = "";
406       break;
407     }
408 
409   return SIM_RC_OK;
410 }
411 
412 /* Add the standard option list to the simulator.  */
413 
414 SIM_RC
standard_install(SIM_DESC sd)415 standard_install (SIM_DESC sd)
416 {
417   SIM_ASSERT (STATE_MAGIC (sd) == SIM_MAGIC_NUMBER);
418   if (sim_add_option_table (sd, NULL, standard_options) != SIM_RC_OK)
419     return SIM_RC_FAIL;
420   STATE_LOAD_AT_LMA_P (sd) = 1;
421   return SIM_RC_OK;
422 }
423 
424 /* Return non-zero if arg is a duplicate argument.
425    If ARG is NULL, initialize.  */
426 
427 #define ARG_HASH_SIZE 97
428 #define ARG_HASH(a) ((256 * (unsigned char) a[0] + (unsigned char) a[1]) % ARG_HASH_SIZE)
429 
430 static int
dup_arg_p(const char * arg)431 dup_arg_p (const char *arg)
432 {
433   int hash;
434   static const char **arg_table = NULL;
435 
436   if (arg == NULL)
437     {
438       if (arg_table == NULL)
439 	arg_table = (const char **) xmalloc (ARG_HASH_SIZE * sizeof (char *));
440       memset (arg_table, 0, ARG_HASH_SIZE * sizeof (char *));
441       return 0;
442     }
443 
444   hash = ARG_HASH (arg);
445   while (arg_table[hash] != NULL)
446     {
447       if (strcmp (arg, arg_table[hash]) == 0)
448 	return 1;
449       /* We assume there won't be more than ARG_HASH_SIZE arguments so we
450 	 don't check if the table is full.  */
451       if (++hash == ARG_HASH_SIZE)
452 	hash = 0;
453     }
454   arg_table[hash] = arg;
455   return 0;
456 }
457 
458 /* Called by sim_open to parse the arguments.  */
459 
460 SIM_RC
sim_parse_args(SIM_DESC sd,char * const * argv)461 sim_parse_args (SIM_DESC sd, char * const *argv)
462 {
463   int c, i, argc, num_opts, save_opterr;
464   char *p, *short_options;
465   /* The `val' option struct entry is dynamically assigned for options that
466      only come in the long form.  ORIG_VAL is used to get the original value
467      back.  */
468   int *orig_val;
469   struct option *lp, *long_options;
470   const struct option_list *ol;
471   const OPTION *opt;
472   OPTION_HANDLER **handlers;
473   sim_cpu **opt_cpu;
474   SIM_RC result = SIM_RC_OK;
475 
476   /* Count the number of arguments.  */
477   argc = countargv (argv);
478 
479   /* Count the number of options.  */
480   num_opts = 0;
481   for (ol = STATE_OPTIONS (sd); ol != NULL; ol = ol->next)
482     for (opt = ol->options; OPTION_VALID_P (opt); ++opt)
483       ++num_opts;
484   for (i = 0; i < MAX_NR_PROCESSORS; ++i)
485     for (ol = CPU_OPTIONS (STATE_CPU (sd, i)); ol != NULL; ol = ol->next)
486       for (opt = ol->options; OPTION_VALID_P (opt); ++opt)
487 	++num_opts;
488 
489   /* Initialize duplicate argument checker.  */
490   (void) dup_arg_p (NULL);
491 
492   /* Build the option table for getopt.  */
493 
494   long_options = NZALLOC (struct option, num_opts + 1);
495   lp = long_options;
496   short_options = NZALLOC (char, num_opts * 3 + 1);
497   p = short_options;
498   handlers = NZALLOC (OPTION_HANDLER *, OPTION_START + num_opts);
499   orig_val = NZALLOC (int, OPTION_START + num_opts);
500   opt_cpu = NZALLOC (sim_cpu *, OPTION_START + num_opts);
501 
502   /* Set '+' as first char so argument permutation isn't done.  This
503      is done to stop getopt_long returning options that appear after
504      the target program.  Such options should be passed unchanged into
505      the program image. */
506   *p++ = '+';
507 
508   for (i = OPTION_START, ol = STATE_OPTIONS (sd); ol != NULL; ol = ol->next)
509     for (opt = ol->options; OPTION_VALID_P (opt); ++opt)
510       {
511 	if (dup_arg_p (opt->opt.name))
512 	  continue;
513 	if (opt->shortopt != 0)
514 	  {
515 	    *p++ = opt->shortopt;
516 	    if (opt->opt.has_arg == required_argument)
517 	      *p++ = ':';
518 	    else if (opt->opt.has_arg == optional_argument)
519 	      { *p++ = ':'; *p++ = ':'; }
520 	    handlers[(unsigned char) opt->shortopt] = opt->handler;
521 	    if (opt->opt.val != 0)
522 	      orig_val[(unsigned char) opt->shortopt] = opt->opt.val;
523 	    else
524 	      orig_val[(unsigned char) opt->shortopt] = opt->shortopt;
525 	  }
526 	if (opt->opt.name != NULL)
527 	  {
528 	    *lp = opt->opt;
529 	    /* Dynamically assign `val' numbers for long options. */
530 	    lp->val = i++;
531 	    handlers[lp->val] = opt->handler;
532 	    orig_val[lp->val] = opt->opt.val;
533 	    opt_cpu[lp->val] = NULL;
534 	    ++lp;
535 	  }
536       }
537 
538   for (c = 0; c < MAX_NR_PROCESSORS; ++c)
539     {
540       sim_cpu *cpu = STATE_CPU (sd, c);
541       for (ol = CPU_OPTIONS (cpu); ol != NULL; ol = ol->next)
542 	for (opt = ol->options; OPTION_VALID_P (opt); ++opt)
543 	  {
544 #if 0 /* Each option is prepended with --<cpuname>- so this greatly cuts down
545 	 on the need for dup_arg_p checking.  Maybe in the future it'll be
546 	 needed so this is just commented out, and not deleted.  */
547 	    if (dup_arg_p (opt->opt.name))
548 	      continue;
549 #endif
550 	    /* Don't allow short versions of cpu specific options for now.  */
551 	    if (opt->shortopt != 0)
552 	      {
553 		sim_io_eprintf (sd, "internal error, short cpu specific option");
554 		result = SIM_RC_FAIL;
555 		break;
556 	      }
557 	    if (opt->opt.name != NULL)
558 	      {
559 		char *name;
560 		*lp = opt->opt;
561 		/* Prepend --<cpuname>- to the option.  */
562 		if (asprintf (&name, "%s-%s", CPU_NAME (cpu), lp->name) < 0)
563 		  {
564 		    sim_io_eprintf (sd, "internal error, out of memory");
565 		    result = SIM_RC_FAIL;
566 		    break;
567 		  }
568 		lp->name = name;
569 		/* Dynamically assign `val' numbers for long options. */
570 		lp->val = i++;
571 		handlers[lp->val] = opt->handler;
572 		orig_val[lp->val] = opt->opt.val;
573 		opt_cpu[lp->val] = cpu;
574 		++lp;
575 	      }
576 	  }
577     }
578 
579   /* Terminate the short and long option lists.  */
580   *p = 0;
581   lp->name = NULL;
582 
583   /* Ensure getopt is initialized.  */
584   optind = 0;
585 
586   /* Do not lot getopt throw errors for us.  But don't mess with the state for
587      any callers higher up by saving/restoring it.  */
588   save_opterr = opterr;
589   opterr = 0;
590 
591   while (1)
592     {
593       int longind, optc;
594 
595       optc = getopt_long (argc, argv, short_options, long_options, &longind);
596       if (optc == -1)
597 	{
598 	  if (STATE_OPEN_KIND (sd) == SIM_OPEN_STANDALONE)
599 	    STATE_PROG_ARGV (sd) = dupargv (argv + optind);
600 	  break;
601 	}
602       if (optc == '?')
603 	{
604 	  /* If getopt rejects a short option, optopt is set to the bad char.
605 	     If it rejects a long option, we have to look at optind.  In the
606 	     short option case, argv could be multiple short options.  */
607 	  const char *badopt;
608 	  char optbuf[3];
609 
610 	  if (optopt)
611 	    {
612 	      sprintf (optbuf, "-%c", optopt);
613 	      badopt = optbuf;
614 	    }
615 	  else
616 	    badopt = argv[optind - 1];
617 
618 	  sim_io_eprintf (sd,
619 			  "%s: unrecognized option '%s'\n"
620 			  "Use --help for a complete list of options.\n",
621 			  STATE_MY_NAME (sd), badopt);
622 
623 	  result = SIM_RC_FAIL;
624 	  break;
625 	}
626 
627       if ((*handlers[optc]) (sd, opt_cpu[optc], orig_val[optc], optarg, 0/*!is_command*/) == SIM_RC_FAIL)
628 	{
629 	  result = SIM_RC_FAIL;
630 	  break;
631 	}
632     }
633 
634   opterr = save_opterr;
635 
636   free (long_options);
637   free (short_options);
638   free (handlers);
639   free (opt_cpu);
640   free (orig_val);
641   return result;
642 }
643 
644 /* Utility of sim_print_help to print a list of option tables.  */
645 
646 static void
print_help(SIM_DESC sd,sim_cpu * cpu,const struct option_list * ol,int is_command)647 print_help (SIM_DESC sd, sim_cpu *cpu, const struct option_list *ol, int is_command)
648 {
649   const OPTION *opt;
650 
651   for ( ; ol != NULL; ol = ol->next)
652     for (opt = ol->options; OPTION_VALID_P (opt); ++opt)
653       {
654 	const int indent = 30;
655 	int comma, len;
656 	const OPTION *o;
657 
658 	if (dup_arg_p (opt->opt.name))
659 	  continue;
660 
661 	if (opt->doc == NULL)
662 	  continue;
663 
664 	if (opt->doc_name != NULL && opt->doc_name [0] == '\0')
665 	  continue;
666 
667 	sim_io_printf (sd, "  ");
668 
669 	comma = 0;
670 	len = 2;
671 
672 	/* list any short options (aliases) for the current OPT */
673 	if (!is_command)
674 	  {
675 	    o = opt;
676 	    do
677 	      {
678 		if (o->shortopt != '\0')
679 		  {
680 		    sim_io_printf (sd, "%s-%c", comma ? ", " : "", o->shortopt);
681 		    len += (comma ? 2 : 0) + 2;
682 		    if (o->arg != NULL)
683 		      {
684 			if (o->opt.has_arg == optional_argument)
685 			  {
686 			    sim_io_printf (sd, "[%s]", o->arg);
687 			    len += 1 + strlen (o->arg) + 1;
688 			  }
689 			else
690 			  {
691 			    sim_io_printf (sd, " %s", o->arg);
692 			    len += 1 + strlen (o->arg);
693 			  }
694 		      }
695 		    comma = 1;
696 		  }
697 		++o;
698 	      }
699 	    while (OPTION_VALID_P (o) && o->doc == NULL);
700 	  }
701 
702 	/* list any long options (aliases) for the current OPT */
703 	o = opt;
704 	do
705 	  {
706 	    const char *name;
707 	    const char *cpu_prefix = cpu ? CPU_NAME (cpu) : NULL;
708 	    if (o->doc_name != NULL)
709 	      name = o->doc_name;
710 	    else
711 	      name = o->opt.name;
712 	    if (name != NULL)
713 	      {
714 		sim_io_printf (sd, "%s%s%s%s%s",
715 			       comma ? ", " : "",
716 			       is_command ? "" : "--",
717 			       cpu ? cpu_prefix : "",
718 			       cpu ? "-" : "",
719 			       name);
720 		len += ((comma ? 2 : 0)
721 			+ (is_command ? 0 : 2)
722 			+ strlen (name));
723 		if (o->arg != NULL)
724 		  {
725 		    if (o->opt.has_arg == optional_argument)
726 		      {
727 			sim_io_printf (sd, "[=%s]", o->arg);
728 			len += 2 + strlen (o->arg) + 1;
729 		      }
730 		    else
731 		      {
732 			sim_io_printf (sd, " %s", o->arg);
733 			len += 1 + strlen (o->arg);
734 		      }
735 		  }
736 		comma = 1;
737 	      }
738 	    ++o;
739 	  }
740 	while (OPTION_VALID_P (o) && o->doc == NULL);
741 
742 	if (len >= indent)
743 	  {
744 	    sim_io_printf (sd, "\n%*s", indent, "");
745 	  }
746 	else
747 	  sim_io_printf (sd, "%*s", indent - len, "");
748 
749 	/* print the description, word wrap long lines */
750 	{
751 	  const char *chp = opt->doc;
752 	  unsigned doc_width = 80 - indent;
753 	  while (strlen (chp) >= doc_width) /* some slack */
754 	    {
755 	      const char *end = chp + doc_width - 1;
756 	      while (end > chp && !isspace (*end))
757 		end --;
758 	      if (end == chp)
759 		end = chp + doc_width - 1;
760 	      /* The cast should be ok - its distances between to
761                  points in a string.  */
762 	      sim_io_printf (sd, "%.*s\n%*s", (int) (end - chp), chp, indent,
763 			     "");
764 	      chp = end;
765 	      while (isspace (*chp) && *chp != '\0')
766 		chp++;
767 	    }
768 	  sim_io_printf (sd, "%s\n", chp);
769 	}
770       }
771 }
772 
773 /* Print help messages for the options.  */
774 
775 void
sim_print_help(SIM_DESC sd,int is_command)776 sim_print_help (SIM_DESC sd, int is_command)
777 {
778   if (STATE_OPEN_KIND (sd) == SIM_OPEN_STANDALONE)
779     sim_io_printf (sd, "Usage: %s [options] program [program args]\n",
780 		   STATE_MY_NAME (sd));
781 
782   /* Initialize duplicate argument checker.  */
783   (void) dup_arg_p (NULL);
784 
785   if (STATE_OPEN_KIND (sd) == SIM_OPEN_STANDALONE)
786     sim_io_printf (sd, "Options:\n");
787   else
788     sim_io_printf (sd, "Commands:\n");
789 
790   print_help (sd, NULL, STATE_OPTIONS (sd), is_command);
791   sim_io_printf (sd, "\n");
792 
793   /* Print cpu-specific options.  */
794   {
795     int i;
796 
797     for (i = 0; i < MAX_NR_PROCESSORS; ++i)
798       {
799 	sim_cpu *cpu = STATE_CPU (sd, i);
800 	if (CPU_OPTIONS (cpu) == NULL)
801 	  continue;
802 	sim_io_printf (sd, "CPU %s specific options:\n", CPU_NAME (cpu));
803 	print_help (sd, cpu, CPU_OPTIONS (cpu), is_command);
804 	sim_io_printf (sd, "\n");
805       }
806   }
807 
808   sim_io_printf (sd, "Note: Depending on the simulator configuration some %ss\n",
809 		 STATE_OPEN_KIND (sd) == SIM_OPEN_STANDALONE ? "option" : "command");
810   sim_io_printf (sd, "      may not be applicable\n");
811 
812   if (STATE_OPEN_KIND (sd) == SIM_OPEN_STANDALONE)
813     {
814       sim_io_printf (sd, "\n");
815       sim_io_printf (sd, "program args    Arguments to pass to simulated program.\n");
816       sim_io_printf (sd, "                Note: Very few simulators support this.\n");
817     }
818 }
819 
820 /* Utility of sim_args_command to find the closest match for a command.
821    Commands that have "-" in them can be specified as separate words.
822    e.g. sim memory-region 0x800000,0x4000
823    or   sim memory region 0x800000,0x4000
824    If CPU is non-null, use its option table list, otherwise use the main one.
825    *PARGI is where to start looking in ARGV.  It is updated to point past
826    the found option.  */
827 
828 static const OPTION *
find_match(SIM_DESC sd,sim_cpu * cpu,char * argv[],int * pargi)829 find_match (SIM_DESC sd, sim_cpu *cpu, char *argv[], int *pargi)
830 {
831   const struct option_list *ol;
832   const OPTION *opt;
833   /* most recent option match */
834   const OPTION *matching_opt = NULL;
835   int matching_argi = -1;
836 
837   if (cpu)
838     ol = CPU_OPTIONS (cpu);
839   else
840     ol = STATE_OPTIONS (sd);
841 
842   /* Skip passed elements specified by *PARGI.  */
843   argv += *pargi;
844 
845   for ( ; ol != NULL; ol = ol->next)
846     for (opt = ol->options; OPTION_VALID_P (opt); ++opt)
847       {
848 	int argi = 0;
849 	const char *name = opt->opt.name;
850 	if (name == NULL)
851 	  continue;
852 	while (argv [argi] != NULL
853 	       && strncmp (name, argv [argi], strlen (argv [argi])) == 0)
854 	  {
855 	    name = &name [strlen (argv[argi])];
856 	    if (name [0] == '-')
857 	      {
858 		/* leading match ...<a-b-c>-d-e-f - continue search */
859 		name ++; /* skip `-' */
860 		argi ++;
861 		continue;
862 	      }
863 	    else if (name [0] == '\0')
864 	      {
865 		/* exact match ...<a-b-c-d-e-f> - better than before? */
866 		if (argi > matching_argi)
867 		  {
868 		    matching_argi = argi;
869 		    matching_opt = opt;
870 		  }
871 		break;
872 	      }
873 	    else
874 	      break;
875 	  }
876       }
877 
878   *pargi = matching_argi;
879   return matching_opt;
880 }
881 
882 static char **
complete_option_list(char ** ret,size_t * cnt,const struct option_list * ol,const char * text,const char * word)883 complete_option_list (char **ret, size_t *cnt, const struct option_list *ol,
884 		      const char *text, const char *word)
885 {
886   const OPTION *opt = NULL;
887   int argi;
888   size_t len = strlen (word);
889 
890   for ( ; ol != NULL; ol = ol->next)
891     for (opt = ol->options; OPTION_VALID_P (opt); ++opt)
892       {
893 	const char *name = opt->opt.name;
894 
895 	/* A long option to match against?  */
896 	if (!name)
897 	  continue;
898 
899 	/* Does this option actually match?  */
900 	if (strncmp (name, word, len))
901 	  continue;
902 
903 	ret = xrealloc (ret, ++*cnt * sizeof (ret[0]));
904 	ret[*cnt - 2] = xstrdup (name);
905       }
906 
907   return ret;
908 }
909 
910 /* All leading text is stored in @text, while the current word being
911    completed is stored in @word.  Trailing text of @word is not.  */
912 
913 char **
sim_complete_command(SIM_DESC sd,const char * text,const char * word)914 sim_complete_command (SIM_DESC sd, const char *text, const char *word)
915 {
916   char **ret = NULL;
917   size_t cnt = 1;
918   sim_cpu *cpu;
919 
920   /* Only complete first word for now.  */
921   if (text != word)
922     return ret;
923 
924   cpu = STATE_CPU (sd, 0);
925   if (cpu)
926     ret = complete_option_list (ret, &cnt, CPU_OPTIONS (cpu), text, word);
927   ret = complete_option_list (ret, &cnt, STATE_OPTIONS (sd), text, word);
928 
929   if (ret)
930     ret[cnt - 1] = NULL;
931   return ret;
932 }
933 
934 SIM_RC
sim_args_command(SIM_DESC sd,const char * cmd)935 sim_args_command (SIM_DESC sd, const char *cmd)
936 {
937   /* something to do? */
938   if (cmd == NULL)
939     return SIM_RC_OK; /* FIXME - perhaps help would be better */
940 
941   if (cmd [0] == '-')
942     {
943       /* user specified -<opt> ... form? */
944       char **argv = buildargv (cmd);
945       SIM_RC rc = sim_parse_args (sd, argv);
946       freeargv (argv);
947       return rc;
948     }
949   else
950     {
951       char **argv = buildargv (cmd);
952       const OPTION *matching_opt = NULL;
953       int matching_argi;
954       sim_cpu *cpu;
955 
956       if (argv [0] == NULL)
957 	{
958 	  freeargv (argv);
959 	  return SIM_RC_OK; /* FIXME - perhaps help would be better */
960 	}
961 
962       /* First check for a cpu selector.  */
963       {
964 	char *cpu_name = xstrdup (argv[0]);
965 	char *hyphen = strchr (cpu_name, '-');
966 	if (hyphen)
967 	  *hyphen = 0;
968 	cpu = sim_cpu_lookup (sd, cpu_name);
969 	if (cpu)
970 	  {
971 	    /* If <cpuname>-<command>, point argv[0] at <command>.  */
972 	    if (hyphen)
973 	      {
974 		matching_argi = 0;
975 		argv[0] += hyphen - cpu_name + 1;
976 	      }
977 	    else
978 	      matching_argi = 1;
979 	    matching_opt = find_match (sd, cpu, argv, &matching_argi);
980 	    /* If hyphen found restore argv[0].  */
981 	    if (hyphen)
982 	      argv[0] -= hyphen - cpu_name + 1;
983 	  }
984 	free (cpu_name);
985       }
986 
987       /* If that failed, try the main table.  */
988       if (matching_opt == NULL)
989 	{
990 	  matching_argi = 0;
991 	  matching_opt = find_match (sd, NULL, argv, &matching_argi);
992 	}
993 
994       if (matching_opt != NULL)
995 	{
996 	  switch (matching_opt->opt.has_arg)
997 	    {
998 	    case no_argument:
999 	      if (argv [matching_argi + 1] == NULL)
1000 		matching_opt->handler (sd, cpu, matching_opt->opt.val,
1001 				       NULL, 1/*is_command*/);
1002 	      else
1003 		sim_io_eprintf (sd, "Command `%s' takes no arguments\n",
1004 				matching_opt->opt.name);
1005 	      break;
1006 	    case optional_argument:
1007 	      if (argv [matching_argi + 1] == NULL)
1008 		matching_opt->handler (sd, cpu, matching_opt->opt.val,
1009 				       NULL, 1/*is_command*/);
1010 	      else if (argv [matching_argi + 2] == NULL)
1011 		matching_opt->handler (sd, cpu, matching_opt->opt.val,
1012 				       argv [matching_argi + 1], 1/*is_command*/);
1013 	      else
1014 		sim_io_eprintf (sd, "Command `%s' requires no more than one argument\n",
1015 				matching_opt->opt.name);
1016 	      break;
1017 	    case required_argument:
1018 	      if (argv [matching_argi + 1] == NULL)
1019 		sim_io_eprintf (sd, "Command `%s' requires an argument\n",
1020 				matching_opt->opt.name);
1021 	      else if (argv [matching_argi + 2] == NULL)
1022 		matching_opt->handler (sd, cpu, matching_opt->opt.val,
1023 				       argv [matching_argi + 1], 1/*is_command*/);
1024 	      else
1025 		sim_io_eprintf (sd, "Command `%s' requires only one argument\n",
1026 				matching_opt->opt.name);
1027 	    }
1028 	  freeargv (argv);
1029 	  return SIM_RC_OK;
1030 	}
1031 
1032       freeargv (argv);
1033     }
1034 
1035   /* didn't find anything that remotly matched */
1036   return SIM_RC_FAIL;
1037 }
1038