Go to the previous, next section.
You can use this manual at your leisure to read all about _GDBN__. However, a handful of commands are enough to get started using the debugger. This chapter illustrates these commands.
In this sample session, we emphasize user input like this: input, to make it easier to pick out from the surrounding output.
_0__
One of the preliminary versions of GNU m4
(a generic macro
processor) exhibits the following bug: sometimes, when we change its
quote strings from the default, the commands used to capture one macro's
definition in another stop working. In the following short m4
session, we define a macro foo
which expands to 0000
; we
then use the m4
built-in defn
to define bar
as the
same thing. However, when we change the open quote string to
<QUOTE>
and the close quote string to <UNQUOTE>
, the same
procedure fails to define a new synonym baz
:
$ cd gnu/m4 $ ./m4 define(foo,0000) foo 0000 define(bar,defn(`foo')) bar 0000 changequote(<QUOTE>,<UNQUOTE>) define(baz,defn(<QUOTE>foo<UNQUOTE>)) baz C-d m4: End of input: 0: fatal error: EOF in string
Let's use _GDBN__ to try to see what's going on.
$ _GDBP__ m4 GDB is free software and you are welcome to distribute copies of it under certain conditions; type "show copying" to see the conditions. There is absolutely no warranty for GDB; type "show warranty" for details. GDB _GDB_VN__, Copyright 1992 Free Software Foundation, Inc... (_GDBP__)
_GDBN__ reads only enough symbol data to know where to find the rest when needed; as a result, the first prompt comes up very quickly. We now tell _GDBN__ to use a narrower display width than usual, so that examples will fit in this manual.
(_GDBP__) set width 70
Let's see how the m4
built-in changequote
works.
Having looked at the source, we know the relevant subroutine is
m4_changequote
, so we set a breakpoint there with _GDBN__'s
break
command.
(_GDBP__) break m4_changequote Breakpoint 1 at 0x62f4: file builtin.c, line 879.
Using the run
command, we start m4
running under _GDBN__
control; as long as control does not reach the m4_changequote
subroutine, the program runs as usual:
(_GDBP__) run Starting program: /work/Editorial/gdb/gnu/m4/m4 define(foo,0000) foo 0000
To trigger the breakpoint, we call changequote
. _GDBN__
suspends execution of m4
, displaying information about the
context where it stops.
changequote(<QUOTE>,<UNQUOTE>) Breakpoint 1, m4_changequote (argc=3, argv=0x33c70) at builtin.c:879 879 if (bad_argc(TOKEN_DATA_TEXT(argv[0]),argc,1,3))
Now we use the command n
(next
) to advance execution to
the next line of the current function.
(_GDBP__) n 882 set_quotes((argc >= 2) ? TOKEN_DATA_TEXT(argv[1])\ : nil,
set_quotes
looks like a promising subroutine. We can go into it
by using the command s
(step
) instead of next
.
step
goes to the next line to be executed in any
subroutine, so it steps into set_quotes
.
(_GDBP__) s set_quotes (lq=0x34c78 "<QUOTE>", rq=0x34c88 "<UNQUOTE>") at input.c:530 530 if (lquote != def_lquote)
The display that shows the subroutine where m4
is now
suspended (and its arguments) is called a stack frame display. It
shows a summary of the stack. We can use the backtrace
command (which can also be spelled bt
), to see where we are
in the stack as a whole: the backtrace
command displays a
stack frame for each active subroutine.
(_GDBP__) bt #0 set_quotes (lq=0x34c78 "<QUOTE>", rq=0x34c88 "<UNQUOTE>") at input.c:530 #1 0x6344 in m4_changequote (argc=3, argv=0x33c70) at builtin.c:882 #2 0x8174 in expand_macro (sym=0x33320) at macro.c:242 #3 0x7a88 in expand_token (obs=0x0, t=209696, td=0xf7fffa30) at macro.c:71 #4 0x79dc in expand_input () at macro.c:40 #5 0x2930 in main (argc=0, argv=0xf7fffb20) at m4.c:195
Let's step through a few more lines to see what happens. The first two
times, we can use `s'; the next two times we use n
to avoid
falling into the xstrdup
subroutine.
(_GDBP__) s 0x3b5c 532 if (rquote != def_rquote) (_GDBP__) s 0x3b80 535 lquote = (lq == nil || *lq == '\0') ? \ def_lquote : xstrdup(lq); (_GDBP__) n 536 rquote = (rq == nil || *rq == '\0') ? def_rquote\ : xstrdup(rq); (_GDBP__) n 538 len_lquote = strlen(rquote);
The last line displayed looks a little odd; let's examine the variables
lquote
and rquote
to see if they are in fact the new left
and right quotes we specified. We can use the command p
(print
) to see their values.
(_GDBP__) p lquote $1 = 0x35d40 "<QUOTE>" (_GDBP__) p rquote $2 = 0x35d50 "<UNQUOTE>"
lquote
and rquote
are indeed the new left and right quotes.
Let's look at some context; we can display ten lines of source
surrounding the current line, with the l
(list
) command.
(_GDBP__) l 533 xfree(rquote); 534 535 lquote = (lq == nil || *lq == '\0') ? def_lquote\ : xstrdup (lq); 536 rquote = (rq == nil || *rq == '\0') ? def_rquote\ : xstrdup (rq); 537 538 len_lquote = strlen(rquote); 539 len_rquote = strlen(lquote); 540 } 541 542 void
Let's step past the two lines that set len_lquote
and
len_rquote
, and then examine the values of those variables.
(_GDBP__) n 539 len_rquote = strlen(lquote); (_GDBP__) n 540 } (_GDBP__) p len_lquote $3 = 9 (_GDBP__) p len_rquote $4 = 7
That certainly looks wrong, assuming len_lquote
and
len_rquote
are meant to be the lengths of lquote
and
rquote
respectively. Let's try setting them to better values.
We can use the p
command for this, since it'll print the value of
any expression--and that expression can include subroutine calls and
assignments.
(_GDBP__) p len_lquote=strlen(lquote) $5 = 7 (_GDBP__) p len_rquote=strlen(rquote) $6 = 9
Let's see if that fixes the problem of using the new quotes with the
m4
built-in defn
. We can allow m4
to continue
executing with the c
(continue
) command, and then try the
example that caused trouble initially:
(_GDBP__) c Continuing. define(baz,defn(<QUOTE>foo<UNQUOTE>)) baz 0000
Success! The new quotes now work just as well as the default ones. The
problem seems to have been just the two typos defining the wrong
lengths. We'll let m4
exit by giving it an EOF as input.
C-d Program exited normally.
The message `Program exited normally.' is from _GDBN__; it
indicates m4
has finished executing. We can end our _GDBN__
session with the _GDBN__ quit
command.
(_GDBP__) quit _1__@end smallexample _fi__(!_BARE__)Go to the previous, next section.