[comp.lang.c] Unix command to insert source line print into executable

rlandsma@bbn.com (Rick Landsman) (11/20/90)

I have received numerous requests to provide the answer when I
find my missing tool that inserts "C" source statements as printf
output into the executable for debugging. Turned out it is a
unix command, "ctrace", available on most unix versions. Below
is the man page for those (and others) that requested this info.


Thanks to Bob Seiler at BBN for helping me solve the puzzle.
regards rick - rlandsman@bbn.com



                                                        ctrace(1)



NAME
     ctrace - C program debugger

SYNTAX
     ctrace [_o_p_t_i_o_n_s] [_f_i_l_e]
     ctc [_o_p_t_i_o_n_s] [_f_i_l_e]
     ctcr [_o_p_t_i_o_n_s] [_f_i_l_e]

DESCRIPTION
     The _c_t_r_a_c_e command allows you to follow the execution of a C
     program, statement by statement.  The _c_t_r_a_c_e command reads
     the C program in _f_i_l_e (or from standard input if you do not
     specify _f_i_l_e) and inserts statements to print both the text
     of each executable statement and the values of all variables
     referenced or modified.  It then writes the modified program
     to the standard output.  You must put the output of _c_t_r_a_c_e
     into a temporary file because the _c_c command does not allow
     the use of a pipe.  You then compile and execute this file.

     As each statement in the program executes it is listed at
     the terminal.  The statement is followed by the name and
     value of any variables referenced or modified in the state-
     ment, which is followed by any output from the statement.
     Loops in the trace output are detected and tracing is
     stopped until the loop is exited or a different sequence of
     statements within the loop is executed.  A warning message
     is printed every 1000 times through the loop to help you
     detect infinite loops.

     The trace output goes to the standard output so you can put
     it into a file for examination with an editor or the _t_a_i_l
     command.

     The _c_t_c command is a shell script that prepares the speci-
     fied C program _f_i_l_e for later execution.  The _c_t_c_r command
     is a shell script that both prepares and executes the speci-
     fied C program _f_i_l_e.

OPTIONS
     The only options you will commonly use are:

     -f _f_u_n_c_t_i_o_n_s        Trace only these _f_u_n_c_t_i_o_n_s.

     -v _f_u_n_c_t_i_o_n_s        Trace all but these _f_u_n_c_t_i_o_n_s.

     You may want to add to the default formats for printing
     variables.  Long and pointer variables are always printed as
     signed integers.  Pointers to character arrays are also
     printed as strings if appropriate.  Char, short, and int
     variables are also printed as signed integers and, if
     appropriate, as characters.  Double variables are printed as
     floating point numbers in scientific notation.



                                                                1






ctrace(1)



     You can request that variables be printed in additional for-
     mats, if appropriate, with these options:

     -e                  Floating point

     -o                  Octal

     -u                  Unsigned

     -x                  Hexadecimal

     These options are used only in special circumstances:

     -l _n                Checks _n consecutively executed state-
                         ments for looping trace output, instead
                         of the default of 20.  Use 0 to get all
                         the trace output from loops.

     -P                  Runs the C preprocessor on the input
                         before tracing it.  You can also use the
                         -D, -I, and -U cc(1) preprocessor
                         options.

     -p _s                Changes the trace print functions from
                         the default of "printf(".  For example,
                         "fprintf(stderr," would send the trace
                         to the standard error output.

     -r _f                Uses file _f in place of the _r_u_n_t_i_m_e._c
                         trace function package.  This lets you
                         change the entire print function,
                         instead of just the name and leading
                         arguments.  For further information, see
                         the -p option.

     -s                  Suppresses redundant trace output from
                         simple assignment statements and string
                         copy function calls.  This option can
                         hide a bug caused by use of the = opera-
                         tor in place of the == operator.

     -t _n                Traces _n variables per statement instead
                         of the default of 10 (the maximum number
                         is 20).  The DIAGNOSTICS section
                         explains when to use this option.

EXAMPLES
     Assume the file _l_c._c contains the following C program:







2






                                                        ctrace(1)



           1 #include <stdio.h>
           2 main() /* count lines in input */
           3 {
           4   int c, nl;
           5
           6   nl = 0;
           7   while ((c = getchar()) != EOF)
           8        if (c = '\n')
           9             ++nl;
          10   printf("%d\n", nl);
          11 }


     When you enter the following commands and test data the pro-
     gram is compiled and executed:

          cc lc.c
          a.out
          1
          <CTRL/D>

     The output of the program is the number 2, which is not
     correct because there is only one line in the test data.
     The error in this program is common, but subtle.  When you
     invoke _c_t_r_a_c_e with the following commands:

          ctrace lc.c >temp.c
          cc temp.c
          a.out

     the output is

           2 main()
           6   nl = 0;
               /* nl == 0 */
           7   while ((c = getchar()) != EOF)

     The program is now waiting for input.  If you enter the same
     test data as before, the output is the following:

               /* c == 49 or '1' */
           8        if (c = '\n')
                    /* c == 10 or '\n' */
           9             ++nl;
                         /* nl == 1 */
           7   while ((c = getchar()) != EOF)
               /* c == 10 or '\n' */
           8        if (c = '\n')
                    /* c == 10 or '\n' */
           9             ++nl;
                         /* nl == 2 */
           7   while ((c = getchar()) != EOF)



                                                                3






ctrace(1)



     If you now enter an end of file character <CTRL/D>, the
     final output is the following:

               /* c == -1 */
          10   printf("%d\n", nl);
               /* nl == 2 */2
                return


     Note that the program output printed at the end of the trace
     line for the nl variable.  Also note the return comment
     added by _c_t_r_a_c_e at the end of the trace output.  This shows
     the implicit return at the terminating brace in the func-
     tion.

     The trace output shows that variable c is assigned the value
     "1" in line 7, but in line 8 it has the value "\n".  Once
     your attention is drawn to this _i_f statement, you realize
     that you used the assignment operator (=) in place of the
     equal operator (==).  You can easily miss this error during
     code reading.

EXECUTION-TIME TRACE CONTROL
     The default operation for _c_t_r_a_c_e is to trace the entire pro-
     gram file, unless you use the -f or -v options to trace
     specific functions.  This does not give you statement by
     statement control of the tracing, nor does it let you turn
     the tracing off and on when executing the traced program.

     You can do both of these by adding _c_t_r_o_f_f and _c_t_r_o_n function
     calls to your program to turn the tracing off and on,
     respectively, at execution time.  Thus, you can code arbi-
     trarily complex criteria for trace control with _i_f state-
     ments, and you can even conditionally include this code
     because _c_t_r_a_c_e defines the CTRACE preprocessor variable.
     For example:

          #ifdef CTRACE
               if (c == '!' && i > 1000)
                    ctron();
          #endif

     You can also turn the trace off and on by setting static
     variable tr_ct_ to 0 and 1, respectively.  This is useful if
     you are using a debugger that cannot call these functions
     directly, such as _a_d_b(1).

RESTRICTIONS
     The _c_t_r_a_c_e command does not know about the components of
     aggregates such as structures, unions, and arrays.  It can-
     not choose a format to print all the components of an aggre-
     gate when an assignment is made to the entire aggregate.



4






                                                        ctrace(1)



     The _c_t_r_a_c_e command may choose to print the address of an
     aggregate or use the wrong format (for example, %e for a
     structure with two integer members) when printing the value
     of an aggregate.

     Pointer values are always treated as pointers to character
     strings.

     The loop trace output elimination is done separately for
     each file of a multi-file program.  This can result in func-
     tions called from a loop still being traced, or the elimina-
     tion of trace output from one function in a file until
     another in the same file is called.

WARNINGS
     You get a _c_t_r_a_c_e syntax error if you omit the semicolon at
     the end of the last element declaration in a structure or
     union, just before the right brace (}).  This is optional in
     some C compilers.

     Defining a function with the same name as a system function
     may cause a syntax error if the number of arguments is
     changed.  Use a different name.

     The _c_t_r_a_c_e command assumes that BADMAG is a preprocessor
     macro, and that EOF and NULL are #defined constants.
     Declaring any of these to be variables, for example, "int
     EOF;", will cause a syntax error.

DIAGNOSTICS
     This section contains diagnostic messages from both _c_t_r_a_c_e
     and _c_c, since the traced code often gets some _c_c warning
     messages.  You can get _c_c error messages in some rare cases,
     all of which can be avoided.

     ctrace Diagnostics

     warning: some variables are not traced in this statement

          Only 10 variables are traced in a statement to prevent
          the C compiler "out of tree space; simplify expression"
          error.  Use the -t option to increase this number.



     warning: statement too long to trace

          This statement is over 400 characters long.  Make sure
          that you are using tabs to indent your code, not
          spaces.





                                                                5






ctrace(1)



     cannot handle preprocessor code, use -P option

          This is usually caused by #ifdef/#endif preprocessor
          statements in the middle of a C statement, or by a
          semicolon at the end of a #define preprocessor state-
          ment.



      'if ... else if' sequence too long

          Split the sequence by removing an else from the middle.



     possible syntax error, try -P option

          Use the -P option to preprocess the _c_t_r_a_c_e input, along
          with any appropriate -D, -I, and -U preprocessor
          options.  If you still get the error message, check the
          Warnings section above.



     cc Diagnostics

     warning: floating point not implemented
     warning: illegal combination of pointer and integer
     warning: statement not reached
     warning: sizeof returns 0

          Ignore these messages.



     compiler takes size of function

          See the _c_t_r_a_c_e "possible syntax error" message above.



     yacc stack overflow

          See the _c_t_r_a_c_e "'if ... else if' sequence too long"
          message above.



     out of tree space; simplify expression

          Use the -t option to reduce the number of traced vari-
          ables per statement from the default of 10.  Ignore the



6






                                                        ctrace(1)



          "ctrace: too many variables to trace" warnings you will
          now get.



     redeclaration of signal

          Either correct this declaration of _s_i_g_n_a_l(3), or remove
          it and #include <signal.h>.



     unimplemented structure assignment

          Use _p_c_c instead of _c_c(1).

FILES
     /usr/bin/ctc             preparation shell script
     /usr/bin/ctcr            preparation and run shell script
     /usr/lib/ctrace/runtime.c          run-time trace package

SEE ALSO
     ctype(3), printf(3s), setjmp(3), signal(3), string(3)
































                                                                7





Test Signature