diff options
Diffstat (limited to 'sim/common/sim-options.c')
-rw-r--r-- | sim/common/sim-options.c | 132 |
1 files changed, 93 insertions, 39 deletions
diff --git a/sim/common/sim-options.c b/sim/common/sim-options.c index 5207457..4f8890b 100644 --- a/sim/common/sim-options.c +++ b/sim/common/sim-options.c @@ -26,10 +26,15 @@ with this program; if not, write to the Free Software Foundation, Inc., #include <strings.h> #endif #endif +#ifdef HAVE_STDLIB_H +#include <stdlib.h> +#endif +#include <ctype.h> #include "libiberty.h" #include "../libiberty/alloca-conf.h" #include "sim-options.h" #include "sim-io.h" +#include "sim-assert.h" /* Add a set of options to the simulator. TABLE is an array of OPTIONS terminated by a NULL `opt.name' entry. @@ -115,7 +120,7 @@ static const OPTION standard_options[] = #endif { {"do-command", required_argument, NULL, OPTION_DO_COMMAND}, - '\0', "COMMAND", "Perform a builtin command", + '\0', "COMMAND", ""/*undocumented*/, standard_option_handler }, { {"help", no_argument, NULL, 'H'}, @@ -126,10 +131,11 @@ static const OPTION standard_options[] = }; static SIM_RC -standard_option_handler (sd, opt, arg) +standard_option_handler (sd, opt, arg, is_command) SIM_DESC sd; int opt; char *arg; + int is_command; { int i,n; @@ -149,6 +155,7 @@ standard_option_handler (sd, opt, arg) return SIM_RC_FAIL; } /* FIXME:wip: Need to set something in STATE_CONFIG. */ + current_target_byte_order = BIG_ENDIAN; } else if (strcmp (arg, "little") == 0) { @@ -158,6 +165,7 @@ standard_option_handler (sd, opt, arg) return SIM_RC_FAIL; } /* FIXME:wip: Need to set something in STATE_CONFIG. */ + current_target_byte_order = LITTLE_ENDIAN; } else { @@ -246,6 +254,7 @@ standard_option_handler (sd, opt, arg) SIM_RC standard_install (SIM_DESC sd) { + SIM_ASSERT (STATE_MAGIC (sd) == SIM_MAGIC_NUMBER); if (sim_add_option_table (sd, standard_options) != SIM_RC_OK) return SIM_RC_FAIL; return SIM_RC_OK; @@ -372,7 +381,7 @@ sim_parse_args (sd, argv) if (optc == '?') return SIM_RC_FAIL; - if ((*handlers[optc]) (sd, orig_val[optc], optarg) == SIM_RC_FAIL) + if ((*handlers[optc]) (sd, orig_val[optc], optarg, 0/*!is_command*/) == SIM_RC_FAIL) return SIM_RC_FAIL; } @@ -408,6 +417,9 @@ sim_print_help (sd) if (opt->doc == NULL) continue; + if (opt->doc_name != NULL && opt->doc_name [0] == '\0') + continue; + sim_io_printf (sd, " "); comma = 0; @@ -442,14 +454,19 @@ sim_print_help (sd) o = opt; do { - if (o->opt.name != NULL) + const char *name; + if (o->doc_name != NULL) + name = o->doc_name; + else + name = o->opt.name; + if (name != NULL) { sim_io_printf (sd, "%s--%s", comma ? ", " : "", - o->opt.name); + name); len += ((comma ? 2 : 0) + 2 - + strlen (o->opt.name)); + + strlen (name)); if (o->arg != NULL) { if (o->opt.has_arg == optional_argument) @@ -515,40 +532,77 @@ sim_args_command (sd, cmd) const struct option_list *ol; const OPTION *opt; char **argv = buildargv (cmd); - for (ol = STATE_OPTIONS (sd); ol != NULL; ol = ol->next) - for (opt = ol->options; opt->opt.name != NULL; ++opt) - { - if (strcmp (argv[0], opt->opt.name) == 0) - { - switch (opt->opt.has_arg) - { - case no_argument: - if (argv[1] == NULL) - opt->handler (sd, opt->opt.val, NULL); - else - sim_io_eprintf (sd, "Command `%s' takes no arguments\n", opt->opt.name); - break; - case optional_argument: - if (argv[1] == NULL) - opt->handler (sd, opt->opt.val, NULL); - else if (argv[2] == NULL) - opt->handler (sd, opt->opt.val, argv[1]); - else - sim_io_eprintf (sd, "Command `%s' requires no more than one argument\n", opt->opt.name); + /* most recent option match */ + const OPTION *matching_opt = NULL; + int matching_argi = -1; + if (argv [0] != NULL) + for (ol = STATE_OPTIONS (sd); ol != NULL; ol = ol->next) + for (opt = ol->options; opt->opt.name != NULL; ++opt) + { + int argi = 0; + const char *name = opt->opt.name; + while (strncmp (name, argv [argi], strlen (argv [argi])) == 0) + { + name = &name [strlen (argv[argi])]; + if (name [0] == '-') + { + /* leading match ...<a-b-c>-d-e-f - continue search */ + name ++; /* skip `-' */ + argi ++; + continue; + } + else if (name [0] == '\0') + { + /* exact match ...<a-b-c-d-e-f> - better than before? */ + if (argi > matching_argi) + { + matching_argi = argi; + matching_opt = opt; + } + break; + } + else break; - case required_argument: - if (argv[1] == NULL) - sim_io_eprintf (sd, "Command `%s' requires an argument\n", opt->opt.name); - else if (argv[2] == NULL) - opt->handler (sd, opt->opt.val, argv[1]); - else - sim_io_eprintf (sd, "Command `%s' requires only one argument\n", opt->opt.name); - } - return SIM_RC_OK; - } - } + } + } + if (matching_opt != NULL) + { + switch (matching_opt->opt.has_arg) + { + case no_argument: + if (argv [matching_argi + 1] == NULL) + matching_opt->handler (sd, matching_opt->opt.val, + NULL, 1/*is_command*/); + else + sim_io_eprintf (sd, "Command `%s' takes no arguments\n", + matching_opt->opt.name); + break; + case optional_argument: + if (argv [matching_argi + 1] == NULL) + matching_opt->handler (sd, matching_opt->opt.val, + NULL, 1/*is_command*/); + else if (argv [matching_argi + 2] == NULL) + matching_opt->handler (sd, matching_opt->opt.val, + argv [matching_argi + 1], 1/*is_command*/); + else + sim_io_eprintf (sd, "Command `%s' requires no more than one argument\n", + matching_opt->opt.name); + break; + case required_argument: + if (argv [matching_argi + 1] == NULL) + sim_io_eprintf (sd, "Command `%s' requires an argument\n", + matching_opt->opt.name); + else if (argv [matching_argi + 2] == NULL) + matching_opt->handler (sd, matching_opt->opt.val, + argv [matching_argi + 1], 1/*is_command*/); + else + sim_io_eprintf (sd, "Command `%s' requires only one argument\n", + matching_opt->opt.name); + } + return SIM_RC_OK; + } } - - /* didn't find anything that matched */ + + /* didn't find anything that remotly matched */ return SIM_RC_FAIL; } |