aboutsummaryrefslogtreecommitdiff
path: root/gdb/tracepoint.c
diff options
context:
space:
mode:
authorMichael Snyder <msnyder@vmware.com>1997-11-24 20:02:28 +0000
committerMichael Snyder <msnyder@vmware.com>1997-11-24 20:02:28 +0000
commit6156728be0612b53eb755d2e07e218cc69f92c6e (patch)
treeaf712a08446a6750938e0460d8f0821e9996cd5d /gdb/tracepoint.c
parent57c0b026557370112658f492511c959aa0a638b4 (diff)
downloadgdb-6156728be0612b53eb755d2e07e218cc69f92c6e.zip
gdb-6156728be0612b53eb755d2e07e218cc69f92c6e.tar.gz
gdb-6156728be0612b53eb755d2e07e218cc69f92c6e.tar.bz2
A ton of my most recent changes, mostly to add post-mortem data analysis,
plus some better error checking and reporting. Still not publically visible (not mentioned in the makefile or changelog).
Diffstat (limited to 'gdb/tracepoint.c')
-rw-r--r--gdb/tracepoint.c1193
1 files changed, 852 insertions, 341 deletions
diff --git a/gdb/tracepoint.c b/gdb/tracepoint.c
index 25bc3c5..15f5394 100644
--- a/gdb/tracepoint.c
+++ b/gdb/tracepoint.c
@@ -27,6 +27,7 @@ Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
#include "value.h"
#include "target.h"
#include "language.h"
+#include "gdb_string.h"
/* readline include files */
#include "readline.h"
@@ -56,6 +57,15 @@ static int tracepoint_count;
/* Number of last traceframe collected. */
static int traceframe_number;
+/* Tracepoint for last traceframe collected. */
+static int tracepoint_number;
+
+/* Symbol for function for last traceframe collected */
+static struct symbol *traceframe_fun;
+
+/* Symtab and line for last traceframe collected */
+static struct symtab_and_line traceframe_sal;
+
/* Utility: returns true if "target remote" */
static int
target_is_remote ()
@@ -78,12 +88,12 @@ trace_error (buf)
{
case '1': /* malformed packet error */
if (*++buf == '0') /* general case: */
- error ("tracepoint.c: badly formed packet.");
+ error ("tracepoint.c: error in outgoing packet.");
else
- error ("tracepoint.c: badly formed packet at field #%d.",
- *buf - '0');
+ error ("tracepoint.c: error in outgoing packet at field #%d.",
+ strtol (buf, NULL, 16));
case '2':
- error ("trace API error '%s'.", buf);
+ error ("trace API error 0x%s.", ++buf);
default:
error ("Target returns error code '%s'.", buf);
}
@@ -120,7 +130,7 @@ trace_receive_regs (buf)
}
/* Utility: wait for reply from stub, while accepting "O" packets */
-static void
+static char *
remote_get_noisy_reply (buf)
char *buf;
{
@@ -142,7 +152,7 @@ remote_get_noisy_reply (buf)
buf[1] != 'K')
remote_console_output (buf + 1); /* 'O' message from stub */
else
- return; /* here's the actual reply */
+ return buf; /* here's the actual reply */
} while (1);
}
@@ -166,6 +176,98 @@ set_traceframe_num (num)
value_from_longest (builtin_type_int, (LONGEST) num));
}
+/* Set tracepoint number to NUM. */
+static void
+set_tracepoint_num (num)
+ int num;
+{
+ tracepoint_number = num;
+ set_internalvar (lookup_internalvar ("tracepoint"),
+ value_from_longest (builtin_type_int, (LONGEST) num));
+}
+
+/* Set externally visible debug variables for querying/printing
+ the traceframe context (line, function, file) */
+
+static void
+set_traceframe_context (trace_pc)
+ CORE_ADDR trace_pc;
+{
+ static struct type *func_string, *file_string;
+ static struct type *func_range, *file_range;
+ static value_ptr func_val, file_val;
+ static struct type *charstar;
+ int len;
+
+ if (charstar == (struct type *) NULL)
+ charstar = lookup_pointer_type (builtin_type_char);
+
+ if (trace_pc == -1) /* cease debugging any trace buffers */
+ {
+ traceframe_fun = 0;
+ traceframe_sal.pc = traceframe_sal.line = 0;
+ traceframe_sal.symtab = NULL;
+ set_internalvar (lookup_internalvar ("trace_func"),
+ value_from_longest (charstar, (LONGEST) 0));
+ set_internalvar (lookup_internalvar ("trace_file"),
+ value_from_longest (charstar, (LONGEST) 0));
+ set_internalvar (lookup_internalvar ("trace_line"),
+ value_from_longest (builtin_type_int, (LONGEST) -1));
+ return;
+ }
+
+ /* save as globals for internal use */
+ traceframe_sal = find_pc_line (trace_pc, 0);
+ traceframe_fun = find_pc_function (trace_pc);
+
+ /* save linenumber as "$trace_line", a debugger variable visible to users */
+ set_internalvar (lookup_internalvar ("trace_line"),
+ value_from_longest (builtin_type_int,
+ (LONGEST) traceframe_sal.line));
+
+ /* save func name as "$trace_func", a debugger variable visible to users */
+ if (traceframe_fun == NULL ||
+ SYMBOL_NAME (traceframe_fun) == NULL)
+ set_internalvar (lookup_internalvar ("trace_func"),
+ value_from_longest (charstar, (LONGEST) 0));
+ else
+ {
+ len = strlen (SYMBOL_NAME (traceframe_fun));
+ func_range = create_range_type (func_range,
+ builtin_type_int, 0, len - 1);
+ func_string = create_array_type (func_string,
+ builtin_type_char, func_range);
+ func_val = allocate_value (func_string);
+ VALUE_TYPE (func_val) = func_string;
+ memcpy (VALUE_CONTENTS_RAW (func_val),
+ SYMBOL_NAME (traceframe_fun),
+ len);
+ func_val->modifiable = 0;
+ set_internalvar (lookup_internalvar ("trace_func"), func_val);
+ }
+
+ /* save file name as "$trace_file", a debugger variable visible to users */
+ if (traceframe_sal.symtab == NULL ||
+ traceframe_sal.symtab->filename == NULL)
+ set_internalvar (lookup_internalvar ("trace_file"),
+ value_from_longest (charstar, (LONGEST) 0));
+ else
+ {
+ len = strlen (traceframe_sal.symtab->filename);
+ file_range = create_range_type (file_range,
+ builtin_type_int, 0, len - 1);
+ file_string = create_array_type (file_string,
+ builtin_type_char, file_range);
+ file_val = allocate_value (file_string);
+ VALUE_TYPE (file_val) = file_string;
+ memcpy (VALUE_CONTENTS_RAW (file_val),
+ traceframe_sal.symtab->filename,
+ len);
+ file_val->modifiable = 0;
+ set_internalvar (lookup_internalvar ("trace_file"), file_val);
+ }
+}
+
/* Low level routine to set a tracepoint.
Returns the tracepoint object so caller can set other things.
Does not set the tracepoint number!
@@ -265,17 +367,17 @@ trace_command (arg, from_tty)
t->number = tracepoint_count;
/* If a canonical line spec is needed use that instead of the
- command string. */
+ command string. */
if (canonical != (char **)NULL && canonical[i] != NULL)
- t->addr_string = canonical[i];
+ t->addr_string = canonical[i];
else if (addr_start)
- t->addr_string = savestring (addr_start, addr_end - addr_start);
+ t->addr_string = savestring (addr_start, addr_end - addr_start);
if (cond_start)
- t->cond_string = savestring (cond_start, cond_end - cond_start);
+ t->cond_string = savestring (cond_start, cond_end - cond_start);
/* Let the UI know of any additions */
if (create_tracepoint_hook)
- create_tracepoint_hook (t);
+ create_tracepoint_hook (t);
}
if (sals.nelts > 1)
@@ -310,18 +412,24 @@ tracepoints_info (tpnum_exp, from_tty)
extern int addressprint; /* print machine addresses? */
if (!found_a_tracepoint++)
- printf_filtered (" *** [info tracepoints header line] ***\n");
-
+ {
+ printf_filtered ("Num Enb ");
+ if (addressprint)
+ printf_filtered ("Address ");
+ printf_filtered ("PassC StepC What\n");
+ }
strcpy (wrap_indent, " ");
if (addressprint)
strcat (wrap_indent, " ");
- printf_filtered ("%-3d %-10s ", t->number,
- t->enabled == enabled ? "enabled" : "disabled");
+ printf_filtered ("%-3d %-3s ", t->number,
+ t->enabled == enabled ? "y" : "n");
if (addressprint)
- { /* FIXME-32x64: need a print_address_numeric with field width */
- printf_filtered ("%s ", local_hex_string_custom ((unsigned long) t->address, "08l"));
- }
+ printf_filtered ("%s ",
+ local_hex_string_custom ((unsigned long) t->address,
+ "08l"));
+ printf_filtered ("%-5d %-5d ", t->pass_count, t->step_count);
+
if (t->source_file)
{
sym = find_pc_function (t->address);
@@ -338,8 +446,6 @@ tracepoints_info (tpnum_exp, from_tty)
else
print_address_symbolic (t->address, gdb_stdout, demangle, " ");
- if (t->pass_count != 0)
- printf_filtered (" passcount = %d", t->pass_count);
printf_filtered ("\n");
if (t->actions)
{
@@ -439,7 +545,7 @@ get_tracepoint_by_number (arg)
char **arg;
{
struct tracepoint *t;
- char *cp;
+ char *end, *copy;
value_ptr val;
int tpnum;
@@ -450,14 +556,17 @@ get_tracepoint_by_number (arg)
tpnum = tracepoint_count;
else if (**arg == '$') /* handle convenience variable */
{
- cp = *arg + 1;
- /* find end of convenience variable name */
- while (**arg && **arg != ' ' && **arg != '\t')
- *arg++;
- /* null-terminate if necessary */
- if (**arg != 0)
- *(*arg++) = 0;
- val = value_of_internalvar (lookup_internalvar (cp));
+ /* Make a copy of the name, so we can null-terminate it
+ to pass to lookup_internalvar(). */
+ end = *arg + 1;
+ while (isalnum(*end) || *end == '_')
+ end++;
+ copy = (char *) alloca (end - *arg);
+ strncpy (copy, *arg + 1, (end - *arg - 1));
+ copy[end - *arg - 1] = '\0';
+ *arg = end;
+
+ val = value_of_internalvar (lookup_internalvar (copy));
if (TYPE_CODE( VALUE_TYPE (val)) != TYPE_CODE_INT)
error ("Convenience variable must have integral type.");
tpnum = (int) value_as_long (val);
@@ -570,6 +679,24 @@ static void read_actions PARAMS((struct tracepoint *));
static void free_actions PARAMS((struct tracepoint *));
static int validate_actionline PARAMS((char *, struct tracepoint *));
+static void
+end_pseudocom (args, from_tty)
+{
+ error ("This command cannot be used at the top level.");
+}
+
+static void
+while_stepping_pseudocom (args, from_tty)
+{
+ error ("This command can only be used in a tracepoint actions list.");
+}
+
+static void
+collect_pseudocom (args, from_tty)
+{
+ error ("This command can only be used in a tracepoint actions list.");
+}
+
static void
trace_actions_command (args, from_tty)
char *args;
@@ -668,6 +795,97 @@ read_actions (t)
discard_cleanups (old_chain);
}
+static char *
+parse_and_eval_memrange (arg, addr, typecode, offset, size)
+ char *arg;
+ CORE_ADDR addr;
+ long *typecode, *size;
+ bfd_signed_vma *offset;
+{
+ char *start = arg;
+ struct expression *exp;
+
+ if (*arg++ != '$' || *arg++ != '(')
+ error ("Internal: bad argument to validate_memrange: %s", start);
+
+ if (*arg == '$') /* register for relative memrange? */
+ {
+ exp = parse_exp_1 (&arg, block_for_pc (addr), 1);
+ if (exp->elts[0].opcode != OP_REGISTER)
+ error ("Bad register operand for memrange: %s", start);
+ if (*arg++ != ',')
+ error ("missing comma for memrange: %s", start);
+ *typecode = exp->elts[1].longconst;
+ }
+ else
+ *typecode = 0;
+
+#if 0
+ /* While attractive, this fails for a number of reasons:
+ 1) parse_and_eval_address does not deal with trailing commas,
+ close-parens etc.
+ 2) There is no safeguard against the user trying to use
+ an out-of-scope variable in an address expression (for instance).
+ 2.5) If you are going to allow semi-arbitrary expressions, you
+ would need to explain which expressions are allowed, and
+ which are not (which would provoke endless questions).
+ 3) If you are going to allow semi-arbitrary expressions in the
+ offset and size fields, then the leading "$" of a register
+ name no longer disambiguates the typecode field.
+ */
+
+ *offset = parse_and_eval_address (arg);
+ if ((arg = strchr (arg, ',')) == NULL)
+ error ("missing comma for memrange: %s", start);
+ else
+ arg++;
+
+ *size = parse_and_eval_address (arg);
+ if ((arg = strchr (arg, ')')) == NULL)
+ error ("missing close-parenthesis for memrange: %s", start);
+ else
+ arg++;
+#else
+#if 0
+ /* This, on the other hand, doesn't work because "-1" is an
+ expression, not an OP_LONG! Fall back to using strtol for now. */
+
+ exp = parse_exp_1 (&arg, block_for_pc (addr), 1);
+ if (exp->elts[0].opcode != OP_LONG)
+ error ("Bad offset operand for memrange: %s", start);
+ *offset = exp->elts[2].longconst;
+
+ if (*arg++ != ',')
+ error ("missing comma for memrange: %s", start);
+
+ exp = parse_exp_1 (&arg, block_for_pc (addr), 1);
+ if (exp->elts[0].opcode != OP_LONG)
+ error ("Bad size operand for memrange: %s", start);
+ *size = exp->elts[2].longconst;
+
+ if (*size <= 0)
+ error ("invalid size in memrange: %s", start);
+
+ if (*arg++ != ')')
+ error ("missing close-parenthesis for memrange: %s", start);
+#else
+ *offset = strtol (arg, &arg, 0);
+ if (*arg++ != ',')
+ error ("missing comma for memrange: %s", start);
+ *size = strtol (arg, &arg, 0);
+ if (*size <= 0)
+ error ("invalid size in memrange: %s", start);
+ if (*arg++ != ')')
+ error ("missing close-parenthesis for memrange: %s", start);
+#endif
+#endif
+ if (info_verbose)
+ printf_filtered ("Collecting memrange: (0x%x,0x%x,0x%x)\n",
+ *typecode, *offset, *size);
+
+ return arg;
+}
+
static enum actionline_type
validate_actionline (line, t)
char *line;
@@ -690,14 +908,24 @@ validate_actionline (line, t)
while (isspace (*p))
p++;
- if (*p == '$' && /* look for special pseudo-symbols */
- ((0 == strncasecmp ("reg", p + 1, 3)) ||
- (0 == strncasecmp ("arg", p + 1, 3)) ||
- (0 == strncasecmp ("loc", p + 1, 3))))
- p = (char *) strchr (p, ',');
+ if (*p == '$') /* look for special pseudo-symbols */
+ {
+ long typecode, size;
+ bfd_signed_vma offset;
+
+ if ((0 == strncasecmp ("reg", p + 1, 3)) ||
+ (0 == strncasecmp ("arg", p + 1, 3)) ||
+ (0 == strncasecmp ("loc", p + 1, 3)))
+ p = strchr (p, ',');
+
+ else if (p[1] == '(') /* literal memrange */
+ p = parse_and_eval_memrange (p, t->address,
+ &typecode, &offset, &size);
+ }
else
{
exp = parse_exp_1 (&p, block_for_pc (t->address), 1);
+
if (exp->elts[0].opcode != OP_VAR_VALUE &&
/*exp->elts[0].opcode != OP_LONG && */
/*exp->elts[0].opcode != UNOP_CAST && */
@@ -1095,17 +1323,30 @@ encode_actions (t, tdp_actions, step_count, stepping_actions)
{
for (i = 0; i < NUM_REGS; i++)
add_register (collect, i);
- action_exp = (char *) strchr (action_exp, ','); /* more? */
+ action_exp = strchr (action_exp, ','); /* more? */
}
else if (0 == strncasecmp ("$arg", action_exp, 4))
{
add_local_symbols (collect, t->address, 'A');
- action_exp = (char *) strchr (action_exp, ','); /* more? */
+ action_exp = strchr (action_exp, ','); /* more? */
}
else if (0 == strncasecmp ("$loc", action_exp, 4))
{
add_local_symbols (collect, t->address, 'L');
- action_exp = (char *) strchr (action_exp, ','); /* more? */
+ action_exp = strchr (action_exp, ','); /* more? */
+ }
+ else if (action_exp[0] == '$' &&
+ action_exp[1] == '(') /* literal memrange */
+ {
+ long typecode, size;
+ bfd_signed_vma offset;
+
+ action_exp = parse_and_eval_memrange (action_exp,
+ t->address,
+ &typecode,
+ &offset,
+ &size);
+ add_memrange (collect, typecode, offset, size);
}
else
{
@@ -1224,6 +1465,9 @@ trace_start_command (args, from_tty)
remote_get_noisy_reply (target_buf);
if (strcmp (target_buf, "OK"))
error ("Bogus reply from target: %s", target_buf);
+ set_traceframe_num (-1); /* all old traceframes invalidated */
+ set_tracepoint_num (-1);
+ set_traceframe_context(-1);
}
else
printf_filtered ("Trace can only be run on remote targets.\n");
@@ -1285,87 +1529,173 @@ trace_limit_command (args, from_tty)
}
static void
+finish_tfind_command (reply, from_tty)
+ char *reply;
+ int from_tty;
+{
+ int target_frameno = -1, target_tracept = -1;
+
+ while (reply && *reply)
+ switch (*reply) {
+ case 'F':
+ if ((target_frameno = strtol (++reply, &reply, 16)) == -1)
+ error ("Target failed to find requested trace frame.");
+ break;
+ case 'T':
+ if ((target_tracept = strtol (++reply, &reply, 16)) == -1)
+ error ("Target failed to find requested trace frame.");
+ break;
+ case 'O': /* "OK"? */
+ if (reply[1] == 'K' && reply[2] == '\0')
+ reply += 2;
+ else
+ error ("Bogus reply from target: %s", reply);
+ break;
+ default:
+ error ("Bogus reply from target: %s", reply);
+ }
+
+ flush_cached_frames ();
+ registers_changed ();
+ select_frame (get_current_frame (), 0);
+ set_traceframe_num (target_frameno);
+ set_tracepoint_num (target_tracept);
+ set_traceframe_context ((get_current_frame ())->pc);
+
+ if (from_tty)
+ print_stack_frame (selected_frame, selected_frame_level, 1);
+}
+
+/* trace_find_command takes a trace frame number n,
+ sends "QTFrame:<n>" to the target,
+ and accepts a reply that may contain several optional pieces
+ of information: a frame number, a tracepoint number, and an
+ indication of whether this is a trap frame or a stepping frame.
+
+ The minimal response is just "OK" (which indicates that the
+ target does not give us a frame number or a tracepoint number).
+ Instead of that, the target may send us a string containing
+ any combination of:
+ F<hexnum> (gives the selected frame number)
+ T<hexnum> (gives the selected tracepoint number)
+ */
+
+static void
trace_find_command (args, from_tty)
char *args;
int from_tty;
{ /* STUB_COMM PART_IMPLEMENTED */
/* this should only be called with a numeric argument */
- int frameno, target_frameno;
- char buf[40], *tmp;
-
+ int frameno = -1;
+ int target_frameno = -1, target_tracept = -1, target_stepfrm = 0;
+ char *tmp;
+
if (target_is_remote ())
{
if (args == 0 || *args == 0)
- frameno = traceframe_number + 1;
- else if (*args != '-' && !isdigit(*args))
- error ("tfind requires a literal (for now): %s rejected.", args);
+ { /* TFIND with no args means find NEXT trace frame. */
+ if (traceframe_number == -1)
+ frameno = 0; /* "next" is first one */
+ else
+ frameno = traceframe_number + 1;
+ }
+ else if (0 == strcmp (args, "-"))
+ {
+ if (traceframe_number == -1)
+ error ("not debugging trace buffer");
+ else if (traceframe_number == 0)
+ error ("already at start of trace buffer");
+
+ frameno = traceframe_number - 1;
+ }
+#if 0
+ else if (0 == strcasecmp (args, "start"))
+ frameno = 0;
+ else if (0 == strcasecmp (args, "none") ||
+ 0 == strcasecmp (args, "end"))
+ frameno = -1;
+#endif
else
- frameno = strtol (args, 0, 0); /* for now, literals only */
+ frameno = parse_and_eval_address (args);
- sprintf (buf, "QTFrame:%x", frameno);
- putpkt (buf);
- remote_get_noisy_reply (target_buf);
+ sprintf (target_buf, "QTFrame:%x", frameno);
+ putpkt (target_buf);
+ tmp = remote_get_noisy_reply (target_buf);
- if (target_buf[0] != 'F')
- error ("Bogus reply from target: %s", target_buf);
- target_frameno = strtol (&target_buf[1], &tmp, 16);
- if (tmp == &target_buf[1])
- error ("Bogus reply from target: %s", target_buf);
- if (target_frameno != frameno)
- warning ("Target replied with different framenumber, %s != %x",
- target_buf, frameno);
-
- set_traceframe_num (target_frameno);
- flush_cached_frames ();
- registers_changed ();
- select_frame (get_current_frame (), 0);
+ if (frameno == -1) /* end trace debugging */
+ { /* hopefully the stub has complied! */
+ if (0 != strcmp (tmp, "F-1"))
+ error ("Bogus response from target: %s", tmp);
+
+ flush_cached_frames ();
+ registers_changed ();
+ select_frame (get_current_frame (), 0);
+ set_traceframe_num (-1);
+ set_tracepoint_num (-1);
+ set_traceframe_context (-1);
+
+ if (from_tty)
+ print_stack_frame (selected_frame, selected_frame_level, 1);
+ }
+ else
+ finish_tfind_command (tmp, from_tty);
}
else
error ("Trace can only be run on remote targets.");
}
static void
+trace_find_end_command (args, from_tty)
+ char *args;
+ int from_tty;
+{
+ trace_find_command ("-1", from_tty);
+}
+
+static void
+trace_find_none_command (args, from_tty)
+ char *args;
+ int from_tty;
+{
+ trace_find_command ("-1", from_tty);
+}
+
+static void
+trace_find_start_command (args, from_tty)
+ char *args;
+ int from_tty;
+{
+ trace_find_command ("0", from_tty);
+}
+
+static void
trace_find_pc_command (args, from_tty)
char *args;
int from_tty;
{ /* STUB_COMM PART_IMPLEMENTED */
CORE_ADDR pc;
int target_frameno;
- char buf[40], *tmp;
+ char *tmp;
if (target_is_remote ())
{
if (args == 0 || *args == 0)
- { /* TFIND PC <no args> is the same as TFIND <no args> */
- trace_find_command (args, from_tty);
- return;
- }
- if (!isdigit(*args))
- error ("tfind pc requires a literal argument (for now): %s rejected.",
- args);
-
- pc = strtol (args, 0, 0); /* for now, literals only */
- sprintf (buf, "QTFrame:pc:%x", pc);
- putpkt (buf);
- remote_get_noisy_reply (target_buf);
+ pc = read_pc (); /* default is current pc */
+ else
+ pc = parse_and_eval_address (args);
- if (target_buf[0] != 'F')
- error ("Bogus reply from target: %s", target_buf);
- target_frameno = strtol (&target_buf[1], &tmp, 16);
- if (tmp == &target_buf[1])
- error ("Bogus reply from target: %s", target_buf);
+ sprintf (target_buf, "QTFrame:pc:%x", pc);
+ putpkt (target_buf);
+ tmp = remote_get_noisy_reply (target_buf);
- set_traceframe_num (target_frameno);
- flush_cached_frames ();
- registers_changed ();
- select_frame (get_current_frame (), 0);
+ finish_tfind_command (tmp, from_tty);
}
else
error ("Trace can only be run on remote targets.");
}
static void
-trace_find_tdp_command (args, from_tty)
+trace_find_tracepoint_command (args, from_tty)
char *args;
int from_tty;
{ /* STUB_COMM PART_IMPLEMENTED */
@@ -1375,145 +1705,203 @@ trace_find_tdp_command (args, from_tty)
if (target_is_remote ())
{
if (args == 0 || *args == 0)
- { /* TFIND TDP <no args> is the same as TFIND <no args> */
- trace_find_command (args, from_tty);
- return;
- }
- if (!isdigit(*args))
- error ("tfind tdp command requires a literal argument (for now): %s",
- args);
-
- tdp = strtol (args, 0, 0); /* for now, literals only */
- sprintf (buf, "QTFrame:tdp:%x", tdp);
- putpkt (buf);
- remote_get_noisy_reply (target_buf);
+ if (tracepoint_number == -1)
+ error ("No current tracepoint -- please supply an argument.");
+ else
+ tdp = tracepoint_number; /* default is current TDP */
+ else
+ tdp = parse_and_eval_address (args);
- if (target_buf[0] != 'F')
- error ("Bogus reply from target: %s", target_buf);
- target_frameno = strtol (&target_buf[1], &tmp, 16);
- if (tmp == &target_buf[1])
- error ("Bogus reply from target: %s", target_buf);
+ sprintf (target_buf, "QTFrame:tdp:%x", tdp);
+ putpkt (target_buf);
+ tmp = remote_get_noisy_reply (target_buf);
- set_traceframe_num (target_frameno);
- flush_cached_frames ();
- registers_changed ();
- select_frame (get_current_frame (), 0);
+ finish_tfind_command (tmp, from_tty);
}
else
error ("Trace can only be run on remote targets.");
}
+/* TFIND LINE command:
+ *
+ * This command will take a sourceline for argument, just like BREAK
+ * or TRACE (ie. anything that "decode_line_1" can handle).
+ *
+ * With no argument, this command will find the next trace frame
+ * corresponding to a source line OTHER THAN THE CURRENT ONE.
+ */
+
static void
-trace_find_range_command (args, from_tty)
+trace_find_line_command (args, from_tty)
char *args;
int from_tty;
{ /* STUB_COMM PART_IMPLEMENTED */
- static CORE_ADDR start, stop;
+ static CORE_ADDR start_pc, end_pc;
+ struct symtabs_and_lines sals;
+ struct symtab_and_line sal;
int target_frameno;
- char buf[50], *tmp;
+ char *tmp;
+ struct cleanup *old_chain;
if (target_is_remote ())
{
- if (args == 0 || *args == 0 || !isdigit(*args))
- { /* XXX FIXME: what should default behavior be? */
- printf_filtered ("Usage: tfind range <address> <address>\n");
- return;
+ if (args == 0 || *args == 0)
+ {
+ sal = find_pc_line ((get_current_frame ())->pc, 0);
+ sals.nelts = 1;
+ sals.sals = (struct symtab_and_line *)
+ xmalloc (sizeof (struct symtab_and_line));
+ sals.sals[0] = sal;
}
-
- start = strtol (args, &args, 0); /* for now, literals only */
- while (args && *args && isspace (*args))
- args++;
-
- if (args == 0 || *args == 0 || !isdigit(*args))
+ else
{
- printf_filtered ("Usage: tfind range <address> <address>\n");
- return;
+ sals = decode_line_spec (args, 1);
+ sal = sals.sals[0];
}
- stop = strtol (args, &args, 0); /* for now, literals only */
-
- sprintf (buf, "QTFrame:range:%x:%x", start, stop);
- putpkt (buf);
- remote_get_noisy_reply (target_buf);
-
- if (target_buf[0] != 'F')
- error ("Bogus reply from target: %s", target_buf);
- target_frameno = strtol (&target_buf[1], &tmp, 16);
- if (tmp == &target_buf[1])
- error ("Bogus reply from target: %s", target_buf);
-
- set_traceframe_num (target_frameno);
- flush_cached_frames ();
- registers_changed ();
- select_frame (get_current_frame (), 0);
+ old_chain = make_cleanup (free, sals.sals);
+ if (sal.symtab == 0)
+ {
+ printf_filtered ("TFIND: No line number information available");
+ if (sal.pc != 0)
+ {
+ /* This is useful for "info line *0x7f34". If we can't tell the
+ user about a source line, at least let them have the symbolic
+ address. */
+ printf_filtered (" for address ");
+ wrap_here (" ");
+ print_address (sal.pc, gdb_stdout);
+ printf_filtered (";\n -- will attempt to find by PC. \n");
+ }
+ else
+ {
+ printf_filtered (".\n");
+ return; /* no line, no PC; what can we do? */
+ }
+ }
+ else if (sal.line > 0
+ && find_line_pc_range (sal, &start_pc, &end_pc))
+ {
+ if (start_pc == end_pc)
+ {
+ printf_filtered ("Line %d of \"%s\"",
+ sal.line, sal.symtab->filename);
+ wrap_here (" ");
+ printf_filtered (" is at address ");
+ print_address (start_pc, gdb_stdout);
+ wrap_here (" ");
+ printf_filtered (" but contains no code.\n");
+ sal = find_pc_line (start_pc, 0);
+ if (sal.line > 0 &&
+ find_line_pc_range (sal, &start_pc, &end_pc) &&
+ start_pc != end_pc)
+ printf_filtered ("Attempting to find line %d instead.\n",
+ sal.line);
+ else
+ error ("Cannot find a good line.");
+ }
+ }
+ else
+ /* Is there any case in which we get here, and have an address
+ which the user would want to see? If we have debugging symbols
+ and no line numbers? */
+ error ("Line number %d is out of range for \"%s\".\n",
+ sal.line, sal.symtab->filename);
+
+ if (args && *args) /* find within range of stated line */
+ sprintf (target_buf, "QTFrame:range:%x:%x", start_pc, end_pc - 1);
+ else /* find OUTSIDE OF range of CURRENT line */
+ sprintf (target_buf, "QTFrame:outside:%x:%x", start_pc, end_pc - 1);
+ putpkt (target_buf);
+ tmp = remote_get_noisy_reply (target_buf);
+
+ finish_tfind_command (tmp, from_tty);
+ do_cleanups (old_chain);
}
else
error ("Trace can only be run on remote targets.");
}
static void
-trace_find_outside_command (args, from_tty)
+trace_find_range_command (args, from_tty)
char *args;
int from_tty;
{ /* STUB_COMM PART_IMPLEMENTED */
- CORE_ADDR start, stop;
+ static CORE_ADDR start, stop;
int target_frameno;
- char buf[50], *tmp;
+ char *tmp;
if (target_is_remote ())
{
- if (args == 0 || *args == 0 || !isdigit(*args))
+ if (args == 0 || *args == 0)
{ /* XXX FIXME: what should default behavior be? */
- printf_filtered ("Usage: tfind outside <address> <address>\n");
+ printf_filtered ("Usage: tfind range <startaddr>,<endaddr>\n");
return;
}
- start = strtol (args, &args, 0);
- while (args && *args && isspace (*args))
- args++;
-
- if (args == 0 || *args == 0 || !isdigit(*args))
+ if (0 != (tmp = strchr (args, ',' )))
{
- printf_filtered ("Usage: tfind outside <address> <address>\n");
- return;
+ *tmp++ = '\0'; /* terminate start address */
+ while (isspace (*tmp))
+ tmp++;
+ start = parse_and_eval_address (args);
+ stop = parse_and_eval_address (tmp);
+ }
+ else
+ { /* no explicit end address? */
+ start = parse_and_eval_address (args);
+ stop = start + 1; /* ??? */
}
- stop = strtol (args, &args, 0);
-
- sprintf (buf, "QTFrame:outside:%x:%x", start, stop);
- putpkt (buf);
- remote_get_noisy_reply (target_buf);
+ sprintf (target_buf, "QTFrame:range:%x:%x", start, stop);
+ putpkt (target_buf);
+ tmp = remote_get_noisy_reply (target_buf);
- if (target_buf[0] != 'F')
- error ("Bogus reply from target: %s", target_buf);
- target_frameno = strtol (&target_buf[1], &tmp, 16);
- if (tmp == &target_buf[1])
- error ("Bogus reply from target: %s", target_buf);
-
- set_traceframe_num (target_frameno);
- flush_cached_frames ();
- registers_changed ();
- select_frame (get_current_frame (), 0);
+ finish_tfind_command (tmp, from_tty);
}
else
error ("Trace can only be run on remote targets.");
}
static void
-trace_display_command (args, from_tty)
+trace_find_outside_command (args, from_tty)
char *args;
int from_tty;
-{ /* STUB_COMM NOT_IMPLEMENTED */
- if (!target_is_remote ())
+{ /* STUB_COMM PART_IMPLEMENTED */
+ CORE_ADDR start, stop;
+ int target_frameno;
+ char *tmp;
+
+ if (target_is_remote ())
{
- printf_filtered ("Trace can only be run on remote targets.\n");
- return;
- }
+ if (args == 0 || *args == 0)
+ { /* XXX FIXME: what should default behavior be? */
+ printf_filtered ("Usage: tfind outside <startaddr>,<endaddr>\n");
+ return;
+ }
- if (args && *args)
- printf_filtered ("Displaying trace as %s.\n", args);
+ if (0 != (tmp = strchr (args, ',' )))
+ {
+ *tmp++ = '\0'; /* terminate start address */
+ while (isspace (*tmp))
+ tmp++;
+ start = parse_and_eval_address (args);
+ stop = parse_and_eval_address (tmp);
+ }
+ else
+ { /* no explicit end address? */
+ start = parse_and_eval_address (args);
+ stop = start + 1; /* ??? */
+ }
+
+ sprintf (target_buf, "QTFrame:outside:%x:%x", start, stop);
+ putpkt (target_buf);
+ tmp = remote_get_noisy_reply (target_buf);
+
+ finish_tfind_command (tmp, from_tty);
+ }
else
- printf_filtered ("Displaying trace.\n");
+ error ("Trace can only be run on remote targets.");
}
static void
@@ -1581,10 +1969,10 @@ scope_info (args, from_tty)
struct symtab_and_line sal;
struct symtabs_and_lines sals;
struct symbol *sym;
+ struct minimal_symbol *msym;
struct block *block;
- char **canonical, *save_args = args, *symname;
+ char **canonical, *symname, *save_args = args;
int i, nsyms, count = 0;
- enum address_class aclass;
if (args == 0 || *args == 0)
error ("requires an argument (function, line or *addr) to define a scope");
@@ -1593,52 +1981,111 @@ scope_info (args, from_tty)
if (sals.nelts == 0)
return; /* presumably decode_line_1 has already warned */
- printf_filtered ("Block for %s", save_args);
- /* Resolve all line numbers to PC's */
- for (i = 0; i < sals.nelts; i++)
- resolve_sal_pc (&sals.sals[i]);
-
+ /* Resolve line numbers to PC */
+ resolve_sal_pc (&sals.sals[0]);
block = block_for_pc (sals.sals[0].pc);
+
while (block != 0)
{
nsyms = BLOCK_NSYMS (block);
for (i = 0; i < nsyms; i++)
{
+ if (count == 0)
+ printf_filtered ("Scope for %s:\n", save_args);
count++;
sym = BLOCK_SYM (block, i);
+ symname = SYMBOL_NAME (sym);
+ if (symname == NULL || *symname == '\0')
+ continue; /* probably botched, certainly useless */
+
+ printf_filtered ("Symbol %s is ", symname);
switch (SYMBOL_CLASS (sym)) {
default:
case LOC_UNDEF: /* messed up symbol? */
- symname = SYMBOL_NAME (sym);
- if (symname && *symname) /* guard against messed up name */
- printf_filtered ("Bogus symbol %s, class %d\n",
- symname, SYMBOL_CLASS (sym));
- else
- printf_filtered ("Completely bogus symbol, class %d.\n",
- SYMBOL_CLASS (sym));
+ printf_filtered ("a bogus symbol, class %d.\n",
+ SYMBOL_CLASS (sym));
count--; /* don't count this one */
continue;
- case LOC_CONST: printf_filtered ("\nConstant "); break;
- case LOC_STATIC: printf_filtered ("\nStatic "); break;
- case LOC_REGISTER: printf_filtered ("\nRegister "); break;
- case LOC_ARG: printf_filtered ("\nArg "); break;
- case LOC_REF_ARG: printf_filtered ("\nReference Arg "); break;
- case LOC_REGPARM: printf_filtered ("\nRegister Arg "); break;
+ case LOC_CONST:
+ printf_filtered ("a constant with value %d (0x%x)",
+ SYMBOL_VALUE (sym), SYMBOL_VALUE (sym));
+ break;
+ case LOC_CONST_BYTES:
+ printf_filtered ("constant bytes: ");
+ if (SYMBOL_TYPE (sym))
+ for (i = 0; i < TYPE_LENGTH (SYMBOL_TYPE (sym)); i++)
+ fprintf_filtered (gdb_stdout, " %02x",
+ (unsigned) SYMBOL_VALUE_BYTES (sym) [i]);
+ break;
+ case LOC_STATIC:
+ printf_filtered ("in static storage at address ");
+ print_address_numeric (SYMBOL_VALUE_ADDRESS (sym), 1, gdb_stdout);
+ break;
+ case LOC_REGISTER:
+ printf_filtered ("a local variable in register $%s",
+ reg_names [SYMBOL_VALUE (sym)]);
+ break;
+ case LOC_ARG:
+ case LOC_LOCAL_ARG:
+ printf_filtered ("an argument at stack/frame offset %ld",
+ SYMBOL_VALUE (sym));
+ break;
+ case LOC_LOCAL:
+ printf_filtered ("a local variable at frame offset %ld",
+ SYMBOL_VALUE (sym));
+ break;
+ case LOC_REF_ARG:
+ printf_filtered ("a reference argument at offset %ld",
+ SYMBOL_VALUE (sym));
+ break;
+ case LOC_REGPARM:
+ printf_filtered ("an argument in register $%s",
+ reg_names[SYMBOL_VALUE (sym)]);
+ break;
case LOC_REGPARM_ADDR:
- printf_filtered ("\nIndirect Register Arg "); break;
- case LOC_LOCAL: printf_filtered ("\nStack Local "); break;
- case LOC_TYPEDEF: printf_filtered ("\nLocal Typedef "); break;
- case LOC_LABEL: printf_filtered ("\nLocal Label "); break;
- case LOC_BLOCK: printf_filtered ("\nLocal Function "); break;
- case LOC_CONST_BYTES: printf_filtered ("\nLoc. Byte Seq. "); break;
- case LOC_LOCAL_ARG: printf_filtered ("\nStack Arg "); break;
- case LOC_BASEREG: printf_filtered ("\nBasereg Local "); break;
- case LOC_BASEREG_ARG: printf_filtered ("\nBasereg Arg "); break;
+ printf_filtered ("the address of an argument, in register $%s",
+ reg_names[SYMBOL_VALUE (sym)]);
+ break;
+ case LOC_TYPEDEF:
+ printf_filtered ("a typedef.\n");
+ continue;
+ case LOC_LABEL:
+ printf_filtered ("a label at address ");
+ print_address_numeric (SYMBOL_VALUE_ADDRESS (sym), 1, gdb_stdout);
+ break;
+ case LOC_BLOCK:
+ printf_filtered ("a function at address ");
+ print_address_numeric (BLOCK_START (SYMBOL_BLOCK_VALUE (sym)), 1,
+ gdb_stdout);
+ break;
+ case LOC_BASEREG:
+ printf_filtered ("a variable at offset %d from register $%s",
+ SYMBOL_VALUE (sym),
+ reg_names [SYMBOL_BASEREG (sym)]);
+ break;
+ case LOC_BASEREG_ARG:
+ printf_filtered ("an argument at offset %d from register $%s",
+ SYMBOL_VALUE (sym),
+ reg_names [SYMBOL_BASEREG (sym)]);
+ break;
case LOC_UNRESOLVED:
- printf_filtered ("\nUnresolved Static "); break;
- case LOC_OPTIMIZED_OUT: printf_filtered ("\nOptimized-Out "); break;
+ msym = lookup_minimal_symbol (SYMBOL_NAME (sym), NULL, NULL);
+ if (msym == NULL)
+ printf_filtered ("Unresolved Static");
+ else
+ {
+ printf_filtered ("static storage at address ");
+ print_address_numeric (SYMBOL_VALUE_ADDRESS (msym), 1,
+ gdb_stdout);
+ }
+ break;
+ case LOC_OPTIMIZED_OUT:
+ printf_filtered ("optimized out.\n");
+ continue;
}
- type_print (SYMBOL_TYPE (sym), SYMBOL_NAME (sym), gdb_stdout, -1);
+ if (SYMBOL_TYPE (sym))
+ printf_filtered (", length %d.\n",
+ TYPE_LENGTH (check_typedef (SYMBOL_TYPE (sym))));
}
if (BLOCK_FUNCTION (block))
break;
@@ -1646,10 +2093,115 @@ scope_info (args, from_tty)
block = BLOCK_SUPERBLOCK (block);
}
if (count <= 0)
- printf_filtered (" contains no locals or arguments.");
- printf_filtered ("\n");
+ printf_filtered ("Scope for %s contains no locals or arguments.\n",
+ save_args);
}
+static void
+replace_comma (comma)
+ char *comma;
+{
+ *comma = ',';
+}
+
+static void
+trace_dump_command (args, from_tty)
+ char *args;
+ int from_tty;
+{
+ struct tracepoint *t;
+ struct action_line *action;
+ char *action_exp, *next_comma;
+ struct cleanup *old_cleanups;
+ int stepping_actions = 0;
+ int stepping_frame = 0;
+
+ if (tracepoint_number == -1)
+ {
+ warning ("No current trace frame.");
+ return;
+ }
+
+ ALL_TRACEPOINTS (t)
+ if (t->number == tracepoint_number)
+ break;
+
+ if (t == NULL)
+ error ("No known tracepoint matches 'current' tracepoint #%d.",
+ tracepoint_number);
+
+ old_cleanups = make_cleanup (null_cleanup, NULL);
+
+ printf_filtered ("Data collected at tracepoint %d, trace frame %d:\n",
+ tracepoint_number, traceframe_number);
+
+ /* The current frame is a trap frame if the frame PC is equal
+ to the tracepoint PC. If not, then the current frame was
+ collected during single-stepping. */
+
+ stepping_frame = (t->address != read_pc());
+
+ for (action = t->actions; action; action = action->next)
+ {
+ action_exp = action->action;
+ while (isspace (*action_exp))
+ action_exp++;
+
+ /* The collection actions to be done while stepping are
+ bracketed by the commands "while-stepping" and "end". */
+
+ if (0 == strncasecmp (action_exp, "while-stepping", 14))
+ stepping_actions = 1;
+ else if (0 == strncasecmp (action_exp, "end", 3))
+ stepping_actions = 0;
+ else if (0 == strncasecmp (action_exp, "collect", 7))
+ {
+ /* Display the collected data.
+ For the trap frame, display only what was collected at the trap.
+ Likewise for stepping frames, display only what was collected
+ while stepping. This means that the two boolean variables,
+ STEPPING_FRAME and STEPPING_ACTIONS should be equal. */
+ if (stepping_frame == stepping_actions)
+ {
+ action_exp += 7;
+ do { /* repeat over a comma-separated list */
+ QUIT;
+ if (*action_exp == ',')
+ action_exp++;
+ while (isspace (*action_exp))
+ action_exp++;
+
+ next_comma = strchr (action_exp, ',');
+ if (next_comma)
+ {
+ make_cleanup (replace_comma, next_comma);
+ *next_comma = '\0';
+ }
+
+ if (0 == strncasecmp (action_exp, "$reg", 4))
+ registers_info (NULL, from_tty);
+ else if (0 == strncasecmp (action_exp, "$loc", 4))
+ locals_info (NULL, from_tty);
+ else if (0 == strncasecmp (action_exp, "$arg", 4))
+ args_info (NULL, from_tty);
+ else
+ {
+ printf_filtered ("%s = ", action_exp);
+ output_command (action_exp, from_tty);
+ printf_filtered ("\n");
+ }
+ if (next_comma)
+ *next_comma = ',';
+ action_exp = next_comma;
+ } while (action_exp && *action_exp == ',');
+ }
+ }
+ }
+ discard_cleanups (old_cleanups);
+}
+
+
+
static struct cmd_list_element *tfindlist;
static struct cmd_list_element *tracelist;
@@ -1658,7 +2210,8 @@ _initialize_tracepoint ()
{
tracepoint_chain = 0;
tracepoint_count = 0;
- traceframe_number = 0;
+ traceframe_number = -1;
+ tracepoint_number = -1;
set_internalvar (lookup_internalvar ("tpnum"),
value_from_longest (builtin_type_int, (LONGEST) 0));
@@ -1681,51 +2234,71 @@ _initialize_tracepoint ()
add_info ("scope", scope_info,
"List the variables local to a scope");
-#if 1
add_cmd ("tracepoints", class_trace, NO_FUNCTION,
- "Tracing program execution without stopping the program.",
+ "Tracing of program execution without stopping the program.",
&cmdlist);
add_info ("tracepoints", tracepoints_info,
- "Display tracepoints, or tracepoint number NUMBER.\n"
- "Convenience variable \"$tpnum\" contains the number of the\n"
- "last tracepoint set.");
+ "Status of tracepoints, or tracepoint number NUMBER.\n\
+Convenience variable \"$tpnum\" contains the number of the\n\
+last tracepoint set.");
add_info_alias ("tp", "tracepoints", 1);
add_com ("save-tracepoints", class_trace, tracepoint_save_command,
- "Save current tracepoint definitions as a script.\n"
- "Use the SOURCE command in another debug session to restore them.");
-
- add_com ("tlimit", class_trace, trace_limit_command,
- "Not sure what this should do yet....");
+ "Save current tracepoint definitions as a script.\n\
+Use the 'source' command in another debug session to restore them.");
- add_com ("tbuffer", class_trace, trace_buff_command,
- "See also 'set trace buffer overflow'.");
+ add_com ("tdump", class_trace, trace_dump_command,
+ "Print everything collected at the current tracepoint.");
- add_prefix_cmd ("tfind", class_trace,
- trace_find_command,
- "Select a trace frame (default by frame number).",
+ add_prefix_cmd ("tfind", class_trace, trace_find_command,
+ "Select a trace frame;\n\
+No argument means forward by one frame; '-' meand backward by one frame.",
&tfindlist, "tfind ", 1, &cmdlist);
add_cmd ("outside", class_trace, trace_find_outside_command,
- "Select a trace frame by falling outside of a PC range",
+ "Select a trace frame whose PC is outside the given \
+range.\nUsage: tfind outside addr1, addr2",
&tfindlist);
add_cmd ("range", class_trace, trace_find_range_command,
- "Select a trace frame by PC range", &tfindlist);
+ "Select a trace frame whose PC is in the given range.\n\
+Usage: tfind range addr1,addr2",
+ &tfindlist);
- add_cmd ("tdp", class_trace, trace_find_tdp_command,
- "Select a trace frame by TDP", &tfindlist);
+ add_cmd ("line", class_trace, trace_find_line_command,
+ "Select a trace frame by source line.\n\
+Argument can be a line number (with optional source file), \n\
+a function name, or '*' followed by an address.\n\
+Default argument is 'the next source line that was traced'.",
+ &tfindlist);
+
+ add_cmd ("tracepoint", class_trace, trace_find_tracepoint_command,
+ "Select a trace frame by tracepoint number.\n\
+Default is the tracepoint for the current trace frame.",
+ &tfindlist);
add_cmd ("pc", class_trace, trace_find_pc_command,
- "Select a trace frame by PC", &tfindlist);
+ "Select a trace frame by PC.\n\
+Default is the current PC, or the PC of the current trace frame.",
+ &tfindlist);
- add_com ("tdisplay", class_trace, trace_display_command,
- "Display the results of a trace");
+ add_cmd ("end", class_trace, trace_find_end_command,
+ "Synonym for 'none'.\n\
+De-select any trace frame and resume 'live' debugging.",
+ &tfindlist);
+
+ add_cmd ("none", class_trace, trace_find_none_command,
+ "De-select any trace frame and resume 'live' debugging.",
+ &tfindlist);
+
+ add_cmd ("start", class_trace, trace_find_start_command,
+ "Select the first trace frame in the trace buffer.",
+ &tfindlist);
add_com ("tstatus", class_trace, trace_status_command,
- "Inquire about trace data collection status.");
+ "Display the status of the current trace data collection.");
add_com ("tstop", class_trace, trace_stop_command,
"Stop trace data collection.");
@@ -1734,134 +2307,72 @@ _initialize_tracepoint ()
"Start trace data collection.");
add_com ("passcount", class_trace, trace_pass_command,
- "Set the passcount for a tracepoint.\n"
- "The trace will end when the tracepoint has been passed "
- "'count' times.\n"
- "Usage: passcount COUNT TPNUM, where TPNUM may also be \"all\",\n"
- "or if omitted refers to the last tracepoint defined.");
+ "Set the passcount for a tracepoint.\n\
+The trace will end when the tracepoint has been passed 'count' times.\n\
+Usage: passcount COUNT TPNUM, where TPNUM may also be \"all\";\n\
+if TPNUM is omitted, passcount refers to the last tracepoint defined.");
+
+ add_com ("end", class_trace, end_pseudocom,
+ "Ends a list of commands or actions.\n\
+Several GDB commands allow you to enter a list of commands or actions.\n\
+Entering \"end\" on a line by itself is the normal way to terminate\n\
+such a list.\n\n\
+Note: the \"end\" command cannot be used at the gdb prompt.");
+
+ add_com ("while-stepping", class_trace, while_stepping_pseudocom,
+ "Specify single-stepping behavior at a tracepoint.\n\
+Argument is number of instructions to trace in single-step mode\n\
+following the tracepoint. This command is normally followed by\n\
+one or more \"collect\" commands, to specify what to collect\n\
+while single-stepping.\n\n\
+Note: this command can only be used in a tracepoint \"actions\" list.");
+
+ add_com ("collect", class_trace, collect_pseudocom,
+ "Specify one or more data items to be collected at a tracepoint.\n\
+Accepts a comma-separated list of (one or more) arguments.\n\
+Things that may be collected include registers, variables, plus\n\
+the following special arguments:\n\
+ $regs -- all registers.\n\
+ $args -- all function arguments.\n\
+ $locals -- all variables local to the block/function scope.\n\
+ $(addr,len) -- a literal memory range.\n\
+ $($reg,addr,len) -- a register-relative literal memory range.\n\n\
+Note: this command can only be used in a tracepoint \"actions\" list.");
add_com ("actions", class_trace, trace_actions_command,
- "Specify the actions to be taken at a tracepoint.\n"
- "Actions can include collection of data, enabling or \n"
- "disabling other tracepoints, or ending the trace.");
+ "Specify the actions to be taken at a tracepoint.\n\
+Tracepoint actions may include collecting of specified data, \n\
+single-stepping, or enabling/disabling other tracepoints, \n\
+depending on target's capabilities.");
add_cmd ("tracepoints", class_trace, delete_trace_command,
- "Delete specified tracepoints; or with no argument, delete all.",
+ "Delete specified tracepoints.\n\
+Arguments are tracepoint numbers, separated by spaces.\n\
+No argument means delete all tracepoints.",
&deletelist);
add_cmd ("tracepoints", class_trace, disable_trace_command,
- "Disable specified tracepoints; or with no argument, disable all.",
+ "Disable specified tracepoints.\n\
+Arguments are tracepoint numbers, separated by spaces.\n\
+No argument means disable all tracepoints.",
&disablelist);
add_cmd ("tracepoints", class_trace, enable_trace_command,
- "Enable specified tracepoints; or with no argument, enable all.",
+ "Enable specified tracepoints.\n\
+Arguments are tracepoint numbers, separated by spaces.\n\
+No argument means enable all tracepoints.",
&enablelist);
add_com ("trace", class_trace, trace_command,
- "Set a tracepoint at a specified line, function, or address.\n"
- "Argument may be a line number, function name, or "
- "'*' plus an address.\n"
- "For a line number or function, trace at the start of its code.\n"
- "If an address is specified, trace at that exact address.");
+ "Set a tracepoint at a specified line or function or address.\n\
+Argument may be a line number, function name, or '*' plus an address.\n\
+For a line number or function, trace at the start of its code.\n\
+If an address is specified, trace at that exact address.\n\n\
+Do \"help tracepoints\" for info on other tracepoint commands.");
add_com_alias ("tp", "trace", class_alias, 0);
add_com_alias ("tr", "trace", class_alias, 1);
add_com_alias ("tra", "trace", class_alias, 1);
add_com_alias ("trac", "trace", class_alias, 1);
-
-
-#else /* command set based on TRACE as a prefix (incomplete) */
- add_cmd ("tracepoints", class_trace, NO_FUNCTION,
- "Tracing program execution without stopping the program.",
- &cmdlist);
-
- add_prefix_cmd ("trace", class_trace, trace_command,
- "prefix for tracing commands",
- &tracelist, "trace ", 1, &cmdlist);
-
- add_cmd ("limit", class_trace, trace_limit_command,
- "Not sure what the hell this does....", &tracelist);
-
- add_cmd ("buffer", class_trace, trace_buff_command,
- "See also 'set trace buffer overflow'.", &tracelist);
-
- add_prefix_cmd ("find", class_trace, trace_find_command,
- "Select a trace frame (default by frame number).",
- &tfindlist, "trace find ", 1, &tracelist);
-
- add_cmd ("outside", class_trace, trace_find_outside_command,
- "Select a tracepoint by falling outside of a PC range.",
- &tfindlist);
-
- add_cmd ("range", class_trace, trace_find_range_command,
- "Select a tracepoint by PC range.",
- &tfindlist);
-
- add_cmd ("pc", class_trace, trace_find_pc_command,
- "Select a tracepoint by PC.",
- &tfindlist);
-
- add_cmd ("display", class_trace, trace_display_command,
- "Display the results of a trace.", &tracelist);
-
- add_cmd ("delete", class_trace, delete_trace_command,
- "Delete some tracepoints; no argument means all.", &tracelist);
-
- add_cmd ("disable", class_trace, disable_trace_command,
- "Disable some tracepoints; no argument means all.", &tracelist);
-
- add_cmd ("enable", class_trace, enable_trace_command,
- "Enable some tracepoints; no argument means all.", &tracelist);
-
- add_cmd ("tracepoints", class_trace, delete_trace_command,
- "Delete some tracepoints; no argument means all.",
- &deletelist);
-
- add_cmd ("tracepoints", class_trace, disable_trace_command,
- "Disable some tracepoints; no argument means all.",
- &disablelist);
-
- add_cmd ("tracepoints", class_trace, enable_trace_command,
- "Enable some tracepoints; no argument means all.",
- &enablelist);
-
- add_cmd ("at", class_trace, trace_command,
- "Set a tracepoint at a specified line or function.\n"
- "Argument may be a line number, function name, or "
- "'*' and an address.\n"
- "For a line number or function, trace from the start of "
- "its code.\n"
- "If an address is specified, trace at that exact address.\n"
- "With no arg, uses current execution address of "
- "selected stack frame.\n"
- "This is useful for breaking on return to a stack frame.",
- &tracelist);
-
- add_cmd ("status", class_trace, trace_status_command,
- "Inquire about trace data collection status.", &tracelist);
-
- add_cmd ("stop", class_trace, trace_stop_command,
- "Stop trace data collection.", &tracelist);
-
- add_cmd ("start", class_trace, trace_start_command,
- "Start trace data collection.", &tracelist);
-
- add_cmd ("info", class_info, tracepoints_info,
- "Status of tracepoints, or tracepoint number NUMBER.\n"
- "The \"Address\" and \"What\" columns indicate the\n"
- "address and file/line number respectively.\n\n"
- "Convenience variable \"$tpnum\" contains the number of the\n"
- "last tracepoint set.",
- &tracelist);
-
- add_info ("tracepoints", tracepoints_info,
- "Status of tracepoints, or tracepoint number NUMBER.\n"
- "The \"Address\" and \"What\" columns indicate the\n"
- "address and file/line number respectively.\n\n"
- "Convenience variable \"$tpnum\" contains the number of the\n"
- "last tracepoint set.");
-
- add_info_alias ("tp", "tracepoints", 1);
-
-#endif
}
+