123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967 |
- /* Print values for GNU debugger gdb.
- Copyright (C) 1986, 1987 Free Software Foundation, Inc.
- GDB is distributed in the hope that it will be useful, but WITHOUT ANY
- WARRANTY. No author or distributor accepts responsibility to anyone
- for the consequences of using it or for whether it serves any
- particular purpose or works at all, unless he says so in writing.
- Refer to the GDB General Public License for full details.
- Everyone is granted permission to copy, modify and redistribute GDB,
- but only under the conditions described in the GDB General Public
- License. A copy of this license is supposed to have been given to you
- along with GDB so you can know your rights and responsibilities. It
- should be in a file named COPYING. Among other things, the copyright
- notice and this notice must be preserved on all copies.
- In other words, go ahead and share GDB, but don't try to stop
- anyone else from sharing it farther. Help stamp out software hoarding!
- */
- #include <stdio.h>
- #include "defs.h"
- #include "initialize.h"
- #include "param.h"
- #include "symtab.h"
- #include "value.h"
- #include "expression.h"
- struct format_data
- {
- int count;
- char format;
- char size;
- };
- /* Last specified output format. */
- static char last_format = 'x';
- /* Last specified examination size. 'b', 'h', 'w' or `q'. */
- static char last_size = 'w';
- /* Default address to examine next. */
- static CORE_ADDR next_address;
- /* Last address examined. */
- static CORE_ADDR last_examine_address;
- /* Contents of last address examined.
- This is not valid past the end of the `x' command! */
- static value last_examine_value;
- void do_displays ();
- void print_address ();
- START_FILE
- /* Decode a format specification. *STRING_PTR should point to it.
- OFORMAT and OSIZE are used as defaults for the format and size
- if none are given in the format specification.
- The structure returned describes all the data
- found in the specification. In addition, *STRING_PTR is advanced
- past the specification and past all whitespace following it. */
- struct format_data
- decode_format (string_ptr, oformat, osize)
- char **string_ptr;
- char oformat;
- char osize;
- {
- struct format_data val;
- register char *p = *string_ptr;
- val.format = oformat;
- val.size = osize;
- val.count = 1;
- if (*p >= '0' && *p <= '9')
- val.count = atoi (p);
- while (*p >= '0' && *p <= '9') p++;
- /* Now process size or format letters that follow. */
- while (1)
- {
- if (*p == 'b' || *p == 'h' || *p == 'w' || *p == 'g')
- val.size = *p++;
- else if (*p >= 'a' && *p <= 'z')
- val.format = *p++;
- else
- break;
- }
- while (*p == ' ' || *p == '\t') p++;
- *string_ptr = p;
- return val;
- }
- /* Print value VAL on stdout according to FORMAT, a letter or 0.
- Do not end with a newline.
- 0 means print VAL according to its own type. */
- static void
- print_formatted (val, format)
- register value val;
- register char format;
- {
- register CORE_ADDR val_long;
- int len = TYPE_LENGTH (VALUE_TYPE (val));
- if (VALUE_LVAL (val) == lval_memory)
- next_address = VALUE_ADDRESS (val) + len;
- if (format && format != 's')
- {
- val_long = value_as_long (val);
- /* If value is unsigned, truncate it in case negative. */
- if (format != 'd')
- {
- if (len == sizeof (char))
- val_long &= (1 << 8 * sizeof(char)) - 1;
- else if (len == sizeof (short))
- val_long &= (1 << 8 * sizeof(short)) - 1;
- }
- }
- switch (format)
- {
- case 's':
- next_address = VALUE_ADDRESS (val)
- + value_print (value_addr (val), stdout);
- break;
- case 'i':
- next_address = VALUE_ADDRESS (val)
- + print_insn (VALUE_ADDRESS (val), stdout);
- break;
- case 'x':
- printf ("0x%x", val_long);
- break;
- case 'd':
- printf ("%d", val_long);
- break;
- case 'u':
- printf ("%u", val_long);
- break;
- case 'o':
- if (val_long)
- printf ("0%o", val_long);
- else
- printf ("0");
- break;
- case 'a':
- print_address (val_long, stdout);
- break;
- case 'c':
- value_print (value_cast (builtin_type_char, val), stdout);
- break;
- case 'f':
- if (TYPE_LENGTH (VALUE_TYPE (val)) == sizeof (float))
- VALUE_TYPE (val) = builtin_type_float;
- if (TYPE_LENGTH (VALUE_TYPE (val)) == sizeof (double))
- VALUE_TYPE (val) = builtin_type_double;
- printf ("%g", value_as_double (val));
- break;
- case 0:
- value_print (val, stdout);
- break;
- default:
- error ("Undefined output format \"%c\".", format);
- }
- }
- /* Specify default address for `x' command.
- `info lines' uses this. */
- void
- set_next_address (addr)
- CORE_ADDR addr;
- {
- next_address = addr;
- /* Make address available to the user as $_. */
- set_internalvar (lookup_internalvar ("_"),
- value_from_long (builtin_type_int, addr));
- }
- /* Print address ADDR symbolically on STREAM.
- First print it as a number. Then perhaps print
- <SYMBOL + OFFSET> after the number. */
- void
- print_address (addr, stream)
- CORE_ADDR addr;
- FILE *stream;
- {
- register int i;
- fprintf (stream, "0x%x", addr);
- i = find_pc_misc_function (addr);
- if (i >= 0)
- if (misc_function_vector[i].address != addr)
- fprintf (stream, " <%s+%d>",
- misc_function_vector[i].name,
- addr - misc_function_vector[i].address);
- else
- fprintf (stream, " <%s>", misc_function_vector[i].name);
- }
- /* Examine data at address ADDR in format FMT.
- Fetch it from memory and print on stdout. */
- static void
- do_examine (fmt, addr)
- struct format_data fmt;
- CORE_ADDR addr;
- {
- register char format = 0;
- register char size;
- register int count = 1;
- struct type *val_type;
- register int i;
- register int maxelts;
- format = fmt.format;
- size = fmt.size;
- count = fmt.count;
- next_address = addr;
- /* String or instruction format implies fetch single bytes
- regardless of the specified size. */
- if (format == 's' || format == 'i')
- size = 'b';
- if (size == 'b')
- val_type = builtin_type_char;
- else if (size == 'h')
- val_type = builtin_type_short;
- else if (size == 'w')
- val_type = builtin_type_long;
- else if (size == 'g')
- val_type = builtin_type_double;
- maxelts = 8;
- if (size == 'w')
- maxelts = 4;
- if (size == 'g')
- maxelts = 2;
- if (format == 's' || format == 'i')
- maxelts = 1;
- /* Print as many objects as specified in COUNT, at most maxelts per line,
- with the address of the next one at the start of each line. */
- while (count > 0)
- {
- print_address (next_address, stdout);
- for (i = maxelts;
- i > 0 && count > 0;
- i--, count--)
- {
- fputc ('\t', stdout);
- /* Note that this sets next_address for the next object. */
- last_examine_address = next_address;
- last_examine_value = value_at (val_type, next_address);
- print_formatted (last_examine_value, format);
- }
- fputc ('\n', stdout);
- fflush (stdout);
- }
- }
- static void
- validate_format (fmt, cmdname)
- struct format_data fmt;
- char *cmdname;
- {
- if (fmt.size != 0)
- error ("Size letters are meaningless in \"%s\" command.", cmdname);
- if (fmt.count != 1)
- error ("Item count other than 1 is meaningless in \"%s\" command.",
- cmdname);
- if (fmt.format == 'i' || fmt.format == 's')
- error ("Format letter \"%c\" is meaningless in \"%s\" command.",
- fmt.format, cmdname);
- }
- static void
- print_command (exp)
- char *exp;
- {
- struct expression *expr;
- register struct cleanup *old_chain = 0;
- register char format = 0;
- register value val;
- struct format_data fmt;
- int histindex;
- int cleanup = 0;
- if (exp && *exp == '/')
- {
- exp++;
- fmt = decode_format (&exp, last_format, 0);
- validate_format (fmt, "print");
- last_format = format = fmt.format;
- }
- if (exp && *exp)
- {
- expr = parse_c_expression (exp);
- old_chain = make_cleanup (free_current_contents, &expr);
- cleanup = 1;
- val = evaluate_expression (expr);
- }
- else
- val = access_value_history (0);
- histindex = record_latest_value (val);
- printf ("$%d = ", histindex);
- print_formatted (val, format);
- printf ("\n");
- if (cleanup)
- do_cleanups (old_chain);
- }
- static void
- output_command (exp)
- char *exp;
- {
- struct expression *expr;
- register struct cleanup *old_chain;
- register char format = 0;
- register value val;
- struct format_data fmt;
- if (exp && *exp == '/')
- {
- exp++;
- fmt = decode_format (&exp, 0, 0);
- validate_format (fmt, "print");
- format = fmt.format;
- }
- expr = parse_c_expression (exp);
- old_chain = make_cleanup (free_current_contents, &expr);
- val = evaluate_expression (expr);
- print_formatted (val, format);
- do_cleanups (old_chain);
- }
- static void
- set_command (exp)
- char *exp;
- {
- struct expression *expr = parse_c_expression (exp);
- register struct cleanup *old_chain
- = make_cleanup (free_current_contents, &expr);
- evaluate_expression (expr);
- do_cleanups (old_chain);
- }
- static void
- address_info (exp)
- char *exp;
- {
- register struct symbol *sym;
- register CORE_ADDR val;
- if (exp == 0)
- error ("Argument required.");
- sym = lookup_symbol (exp, get_selected_block (), VAR_NAMESPACE);
- if (sym == 0)
- {
- register int i;
- for (i = 0; i < misc_function_count; i++)
- if (!strcmp (misc_function_vector[i].name, exp))
- break;
- if (i < misc_function_count)
- printf ("Symbol \"%s\" is at 0x%x in a file compiled without -g.\n",
- exp, misc_function_vector[i].address);
- else
- error ("No symbol \"%s\" in current context.", exp);
- return;
- }
- printf ("Symbol \"%s\" is ", SYMBOL_NAME (sym));
- val = SYMBOL_VALUE (sym);
- switch (SYMBOL_CLASS (sym))
- {
- case LOC_CONST:
- printf ("constant");
- break;
- case LOC_LABEL:
- printf ("a label at address 0x%x", val);
- break;
- case LOC_REGISTER:
- printf ("a variable in register %s", reg_names[val]);
- break;
- case LOC_STATIC:
- printf ("static at address 0x%x", val);
- break;
- case LOC_ARG:
- printf ("an argument at offset %d", val);
- break;
- case LOC_LOCAL:
- printf ("a local variable at frame offset %d", val);
- break;
- case LOC_TYPEDEF:
- printf ("a typedef");
- break;
- case LOC_BLOCK:
- printf ("a function at address 0x%x", val);
- break;
- }
- printf (".\n");
- }
- static void
- x_command (exp, from_tty)
- char *exp;
- int from_tty;
- {
- struct expression *expr;
- struct format_data fmt;
- struct cleanup *old_chain;
- fmt.format = last_format;
- fmt.size = last_size;
- fmt.count = 1;
- if (exp && *exp == '/')
- {
- exp++;
- fmt = decode_format (&exp, last_format, last_size);
- last_size = fmt.size;
- last_format = fmt.format;
- }
- /* If we have an expression, evaluate it and use it as the address. */
- if (exp != 0 && *exp != 0)
- {
- expr = parse_c_expression (exp);
- /* Cause expression not to be there any more
- if this command is repeated with Newline.
- But don't clobber a user-defined command's definition. */
- if (from_tty)
- *exp = 0;
- old_chain = make_cleanup (free_current_contents, &expr);
- next_address = value_as_long (evaluate_expression (expr));
- do_cleanups (old_chain);
- }
- do_examine (fmt, next_address);
- /* Make last address examined available to the user as $_. */
- set_internalvar (lookup_internalvar ("_"),
- value_from_long (builtin_type_int, last_examine_address));
- /* Make contents of last address examined available to the user as $__. */
- set_internalvar (lookup_internalvar ("__"), last_examine_value);
- }
- /* Commands for printing types of things. */
- static void
- whatis_command (exp)
- char *exp;
- {
- struct expression *expr;
- register value val;
- register struct cleanup *old_chain;
- if (exp)
- {
- expr = parse_c_expression (exp);
- old_chain = make_cleanup (free_current_contents, &expr);
- val = evaluate_type (expr);
- }
- else
- val = access_value_history (0);
- printf ("type = ");
- type_print (VALUE_TYPE (val), "", stdout, 1);
- printf ("\n");
- if (exp)
- do_cleanups (old_chain);
- }
- static void
- ptype_command (typename)
- char *typename;
- {
- register char *p = typename;
- register int len;
- extern struct block *get_current_block ();
- register struct block *b
- = (have_inferior_p () || have_core_file_p ()) ? get_current_block () : 0;
- register struct type *type;
- if (typename == 0)
- error_no_arg ("type name");
- while (*p && *p != ' ' && *p != '\t') p++;
- len = p - typename;
- while (*p == ' ' || *p == '\t') p++;
- if (len == 6 && !strncmp (typename, "struct", 6))
- type = lookup_struct (p, b);
- else if (len == 5 && !strncmp (typename, "union", 5))
- type = lookup_union (p, b);
- else if (len == 4 && !strncmp (typename, "enum", 4))
- type = lookup_enum (p, b);
- else
- {
- type = lookup_typename (typename, b, 1);
- if (type == 0)
- {
- register struct symbol *sym
- = lookup_symbol (typename, b, STRUCT_NAMESPACE);
- if (sym == 0)
- error ("No type named %s.", typename);
- printf ("No type named %s, but there is a ",
- typename);
- switch (TYPE_CODE (SYMBOL_TYPE (sym)))
- {
- case TYPE_CODE_STRUCT:
- printf ("struct");
- break;
- case TYPE_CODE_UNION:
- printf ("union");
- break;
- case TYPE_CODE_ENUM:
- printf ("enum");
- }
- printf (" %s. Type \"help ptype\".\n", typename);
- type = SYMBOL_TYPE (sym);
- }
- }
- type_print (type, "", stdout, 1);
- printf ("\n");
- }
- struct display
- {
- /* Chain link to next auto-display item. */
- struct display *next;
- /* Expression to be evaluated and displayed. */
- struct expression *exp;
- /* Item number of this auto-display item. */
- int number;
- /* Display format specified. */
- struct format_data format;
- /* Block in which expression is to be evaluated. */
- struct block *block;
- };
- /* Chain of expressions whose values should be displayed
- automatically each time the program stops. */
- static struct display *display_chain;
- static int display_number;
- /* Add an expression to the auto-display chain.
- Specify the expression. */
- static void
- display_command (exp)
- char *exp;
- {
- struct format_data fmt;
- register struct expression *expr;
- register struct display *new;
- if (exp == 0)
- {
- do_displays ();
- return;
- }
- if (*exp == '/')
- {
- exp++;
- fmt = decode_format (&exp, 0, 0);
- if (fmt.size && fmt.format == 0)
- fmt.format = 'x';
- if (fmt.format == 'i' || fmt.format == 's')
- fmt.size = 'b';
- }
- else
- {
- fmt.format = 0;
- fmt.size = 0;
- fmt.count = 0;
- }
- expr = parse_c_expression (exp);
- new = (struct display *) xmalloc (sizeof (struct display));
- new->exp = expr;
- new->next = display_chain;
- new->number = ++display_number;
- new->format = fmt;
- display_chain = new;
- dont_repeat ();
- }
- static void
- free_display (d)
- struct display *d;
- {
- free (d->exp);
- free (d);
- }
- /* Clear out the display_chain.
- Done when new symtabs are loaded, since this invalidates
- the types stored in many expressions. */
- void
- clear_displays ()
- {
- register struct display *d;
- while (d = display_chain)
- {
- free (d->exp);
- display_chain = d->next;
- free (d);
- }
- }
- /* Delete some values from the auto-display chain.
- Specify the element numbers. */
- static void
- undisplay_command (args)
- char *args;
- {
- register char *p = args;
- register char *p1;
- register int num;
- register struct display *d, *d1;
- if (args == 0)
- {
- if (query ("Delete all auto-display expressions? "))
- clear_displays ();
- dont_repeat ();
- return;
- }
- while (*p)
- {
- p1 = p;
- while (*p1 >= '0' && *p1 <= '9') p1++;
- if (*p1 && *p1 != ' ' && *p1 != '\t')
- error ("Arguments must be display numbers.");
- num = atoi (p);
- if (display_chain->number == num)
- {
- d1 = display_chain;
- display_chain = d1->next;
- free_display (d1);
- }
- else
- for (d = display_chain; ; d = d->next)
- {
- if (d->next == 0)
- error ("No display number %d.", num);
- if (d->next->number == num)
- {
- d1 = d->next;
- d->next = d1->next;
- free_display (d1);
- break;
- }
- }
- p = p1;
- while (*p == ' ' || *p == '\t') p++;
- }
- dont_repeat ();
- }
- /* Display all of the values on the auto-display chain. */
- void
- do_displays ()
- {
- register struct display *d;
- for (d = display_chain; d; d = d->next)
- {
- printf ("%d: ", d->number);
- if (d->format.size)
- {
- printf ("x/");
- if (d->format.count != 1)
- printf ("%d", d->format.count);
- printf ("%c", d->format.format);
- if (d->format.format != 'i' && d->format.format != 's')
- printf ("%c", d->format.size);
- printf (" ");
- print_expression (d->exp, stdout);
- if (d->format.count != 1)
- printf ("\n");
- else
- printf (" ");
- do_examine (d->format,
- value_as_long (evaluate_expression (d->exp)));
- }
- else
- {
- if (d->format.format)
- printf ("/%c ", d->format.format);
- print_expression (d->exp, stdout);
- printf (" = ");
- print_formatted (evaluate_expression (d->exp), d->format.format);
- printf ("\n");
- }
- fflush (stdout);
- }
- }
- static void
- display_info ()
- {
- register struct display *d;
- if (!display_chain)
- printf ("There are no auto-display expressions now.\n");
- else
- printf ("Auto-display expressions now in effect:\n");
- for (d = display_chain; d; d = d->next)
- {
- printf ("%d: ", d->number);
- if (d->format.size)
- printf ("/%d%c%c ", d->format.count, d->format.size,
- d->format.format);
- else if (d->format.format)
- printf ("/%c ", d->format.format);
- print_expression (d->exp, stdout);
- printf ("\n");
- fflush (stdout);
- }
- }
- /* Print the value in stack frame FRAME of a variable
- specified by a struct symbol. */
- void
- print_variable_value (var, frame, stream)
- struct symbol *var;
- CORE_ADDR frame;
- FILE *stream;
- {
- char *space = (char *) alloca (TYPE_LENGTH (SYMBOL_TYPE (var)));
- value val = read_var_value (var, frame);
- value_print (val, stream);
- }
- /* Print the arguments of a stack frame, given the function FUNC
- running in that frame (as a symbol), the address of the arglist,
- and the number of args according to the stack frame (or -1 if unknown). */
- static void print_frame_nameless_args ();
- print_frame_args (func, addr, num, stream)
- struct symbol *func;
- register CORE_ADDR addr;
- int num;
- FILE *stream;
- {
- struct block *b;
- int nsyms = 0;
- register int i;
- register int last_offset = FRAME_ARGS_SKIP;
- register struct symbol *sym, *nextsym;
- register value val;
- if (func)
- {
- b = SYMBOL_BLOCK_VALUE (func);
- nsyms = BLOCK_NSYMS (b);
- }
- while (1)
- {
- /* Find first arg that is not before LAST_OFFSET. */
- nextsym = 0;
- for (i = 0; i < nsyms; i++)
- {
- QUIT;
- sym = BLOCK_SYM (b, i);
- if (SYMBOL_CLASS (sym) == LOC_ARG
- && SYMBOL_VALUE (sym) >= last_offset
- && (nextsym == 0
- || SYMBOL_VALUE (sym) < SYMBOL_VALUE (nextsym)))
- nextsym = sym;
- }
- if (nextsym == 0)
- break;
- sym = nextsym;
- /* Print any nameless args between the last arg printed
- and the next arg. */
- if (last_offset != (SYMBOL_VALUE (sym) / sizeof (int)) * sizeof (int))
- print_frame_nameless_args (addr, last_offset, SYMBOL_VALUE (sym),
- stream);
- /* Print the next arg. */
- val = value_at (SYMBOL_TYPE (sym), addr + SYMBOL_VALUE (sym));
- if (SYMBOL_VALUE (sym) > FRAME_ARGS_SKIP)
- fprintf (stream, ", ");
- fprintf (stream, "%s=", SYMBOL_NAME (sym));
- value_print (val, stream);
- last_offset = SYMBOL_VALUE (sym) + TYPE_LENGTH (SYMBOL_TYPE (sym));
- /* Round up address of next arg to multiple of size of int. */
- last_offset
- = ((last_offset + sizeof (int) - 1) / sizeof (int)) * sizeof (int);
- }
- if (num >= 0 && num * sizeof (int) + FRAME_ARGS_SKIP > last_offset)
- print_frame_nameless_args (addr, last_offset,
- num * sizeof (int) + FRAME_ARGS_SKIP, stream);
- }
- static void
- print_frame_nameless_args (argsaddr, start, end, stream)
- CORE_ADDR argsaddr;
- int start;
- int end;
- FILE *stream;
- {
- while (start < end)
- {
- QUIT;
- if (start != FRAME_ARGS_SKIP)
- fprintf (stream, ", ");
- fprintf (stream, "%d",
- read_memory_integer (argsaddr + start, sizeof (int)));
- start += sizeof (int);
- }
- }
- static
- initialize ()
- {
- add_info ("address", address_info,
- "Describe where variable VAR is stored.");
- add_com ("x", class_vars, x_command,
- "Examine memory: x/FMT ADDRESS.\n\
- ADDRESS is an expression for the memory address to examine.\n\
- FMT is a repeat count followed by a format letter and a size letter.\n\
- Format letters are o(octal), x(hex), d(decimal), u(unsigned decimal),\n\
- f(float), a(address), i(instruction), c(char) and s(string).\n\
- Size letters are b(byte), h(halfword), w(word), g(giant, 8 bytes).\n\
- g is meaningful only with f, for type double.\n\
- The specified number of objects of the specified size are printed\n\
- according to the format.\n\n\
- Defaults for format and size letters are those previously used.\n\
- Default count is 1. Default address is following last thing printed\n\
- with this command or \"print\".");
- add_com ("ptype", class_vars, ptype_command,
- "Print definition of type TYPE.\n\
- Argument may be a type name defined by typedef, or \"struct STRUCTNAME\"\n\
- or \"union UNIONNAME\" or \"enum ENUMNAME\".\n\
- The selected stack frame's lexical context is used to look up the name.");
- add_com ("whatis", class_vars, whatis_command,
- "Print data type of expression EXP.");
- add_info ("display", display_info,
- "Expressions to display when program stops, with code numbers.");
- add_com ("undisplay", class_vars, undisplay_command,
- "Cancel some expressions to be displayed whenever program stops.\n\
- Arguments are the code numbers of the expressions to stop displaying.\n\
- No argument means cancel all automatic-display expressions.\n\
- Do \"info display\" to see current list of code numbers.");
- add_com ("display", class_vars, display_command,
- "Print value of expression EXP each time the program stops.\n\
- /FMT may be used before EXP as in the \"print\" command.\n\
- /FMT \"i\" or \"s\" or including a size-letter is allowed,\n\
- as in the \"x\" command, and then EXP is used to get the address to examine\n\
- and examining is done as in the \"x\" command.\n\n\
- With no argument, display all currently requested auto-display expressions.\n\
- Use \"undisplay\" to cancel display requests previously made.");
- add_com ("output", class_vars, output_command,
- "Like \"print\" but don't put in value history and don't print newline.\n\
- This is useful in user-defined commands.");
- add_com ("set", class_vars, set_command,
- "Perform an assignment VAR = EXP. You must type the \"=\".\n\
- VAR may be a debugger \"convenience\" variables (names starting with $),\n\
- a register (a few standard names starting with $), or an actual variable\n\
- in the program being debugger. EXP is any expression.");
- add_com ("print", class_vars, print_command,
- concat ("Print value of expression EXP.\n\
- Variables accessible are those of the lexical environment of the selected\n\
- stack frame, plus all those whose scope is global or an entire file.\n\
- \n\
- $NUM gets previous value number NUM. $ and $$ are the last two values.\n\
- $$NUM refers to NUM'th value back from the last one.\n\
- Names starting with $ refer to registers (with the values they would have\n\
- if the program were to return to the stack frame now selected, restoring\n\
- all registers saved by frames farther in) or else to debugger\n\
- \"convenience\" variables (any such name not a known register).\n\
- Use assignment expressions to give values to convenience variables.\n",
- "\n\
- \{TYPE}ADREXP refers to a datum of data type TYPE, located at address ADREXP.\n\
- @ is a binary operator for treating consecutive data objects\n\
- anywhere in memory as an array. FOO@NUM gives an array whose first\n\
- element is FOO, whose second element is stored in the space following\n\
- where FOO is stored, etc. FOO must be an expression whose value\n\
- resides in memory.\n",
- "\n\
- EXP may be preceded with /FMT, where FMT is a format letter\n\
- but no count or size letter (see \"x\" command)."));
- add_com_alias ("p", "print", class_vars, 1);
- }
- END_FILE
|