Chapter 7. OTHER TOOLS

OTHER TOOLS

Becoming proficient at debugging code doesn't end with learning to use a debugger like GDB—it begins there. There are a variety of other tools, both free and commercial, that also help to prevent, detect, and eliminate bugs in code. The savvy programmer keeps several of them in his or her bag of tricks, understands what each is good for, and recognizes when to use one of them to save time and effort if bugs appear.

So far we've focused our attention on using symbolic debuggers, but now we'd like to broaden our coverage to other aspects of debugging, including defensive programming. This chapter is devoted to some of the tools and techniques other than GDB that you may find useful, both to prevent bugs from arising in the first place and to find and fix them when they do.

The best debugging method is to not make programming errors to begin with! Simply making good use of an editor that has support for programming is one of the most overlooked aspects of "pre-debugging."

If you spend a lot of time coding, we urge you to think carefully about your choice of editor and to learn as much as you possibly can about the editor you will use. There are two main reasons. First, becoming proficient with a powerful editor decreases the time it takes you to write code. Editors that have specialized features, like automatic indentation, word completion, and global symbol lookup are a boon for programmers, and there are several to choose from. Second, a good editor can actually help the coder catch certain kinds of bugs as the code is being written. That's what this section is all about.

Both authors of this book use Vim for programming, so that's what we'll focus on. However, all the popular editors have similar, if not identical, feature sets. If Vim is enhanced to provide a useful feature not currently in Emacs, the community of Emacs developers would quickly rectify the situation, and vice versa. Therefore, although we give specifics for Vim, most of what we talk about applies to other excellent editors like Emacs as well.

Vim employs syntax highlighting to display portions of a program file with different colors or fonts, so that code elements like keywords, type identifiers, local variables, and preprocessor directives each have their own color and font scheme. The editor selects the color and font scheme by looking at the filename extension to determine the language you're using. For instance, if the filename ends with .pl (indicating a Perl script), occurrences of the word die (which is the name of a Perl function) are highlighted, whereas if the filename ends in .c, they are not.

A better name for syntax highlighting would be lexical highlighting, because the editor generally doesn't analyze syntax too closely. It can't tell you that you've provided the wrong number of arguments or arguments of the wrong type in a function call. Instead, it only understands (for example) that words like bless and foreach are Perl keywords and that fmt and dimension are Fortran keywords and displays them accordingly.

Even so, syntax highlighting is still very useful for catching simple but easy-to-make errors. For instance, on our computer the default color for type identifiers, like the FILE or float keywords in C, is green. Once your eye becomes trained to the font and color scheme, you'll catch the color inconsistency that arises when a type name is misspelled and automatically correct the mistake without having to go through a needless compile cycle.

An example of using syntax highlighting to check keywords that we (the authors) enjoy occurs with makefiles. The patsubst keyword is a very useful text search-and-replace command for makefiles. One of its most common uses is to generate a list of .o files from the .c files of a project's source code:

TARGET = CoolApplication
OBJS = $(patsubst %.c, %.o, $(wildcard *.c))

$(TARGET): $(OBJS)

One of the authors can never remember whether it's patsubst, pathsubst, or patsub. Knowing that makefile keywords are displayed in a light color (yellow), can you figure out which version of the line shown below is incorrect? Even if you don't know how to write makefiles, syntax highlighting alone should make it clear! [16]


Furthermore, here's an example where syntax highlighting is a bit smarter. The figure below has an honest-to-goodness syntax error. Try to find, without thinking too much, what (or at least where) the error is, based on the colors alone:


and here's another illustration of a similar error. Try to let the color guide your eye to the error.


You may find that certain colors in the syntax highlighting scheme are hard to read. If so, you can turn off the highlighting by typing the following:

: syntax off

The command to turn it back on again is of course

: syntax on

A better option would be to modify the syntax file to use a different, better color for that type of keyword, but that's beyond the scope of our discussion here.

Unbalanced bracket errors are extremely common and can be very difficult to catch. Consider the following code:

mytype *myvar;
if ((myvar = (mytype *)malloc(sizeof(mytype))) == NULL) {
   exit(-1);
}

Quickly: Are the parentheses balanced?[17] Have you ever had to track down unbalanced brackets in long blocks of code with lots of conditionals? Or tried to use TEX? (We shudder to think of some of our past LATEX files with missing braces!) Then you must agree with us that that's exactly what computers are for—to relieve us of such tedious work! Vim has some great features that can help.

  • Whenever you type a bracket at the keyboard, Vim's showmatch option makes Vim momentarily place the cursor over the matching bracket (if the matching bracket exists and is visible on the screen). You can even control how long the cursor lingers on the matching bracket by setting the matchtime variable, which specifies this duration in tenths of seconds.

  • Typing the percent symbol when the cursor is on a bracket will move the cursor to the bracket's mate. This command is a great way to track down unbalanced brackets.

  • When you place the cursor on a bracket, Vim will highlight its mate, as shown in Figure ??. This is also a great way to track down unbalanced bracket problems.

The showmatch option is useful when you're programming, but it can otherwise be annoying. You can use autocommands to set this option only when you program. For example, to set showmatch only for editing sessions with C/C++ source code files, you can put lines like these in your .vimrc file (see the Vim helpfiles for more information):

au BufNewFile,BufRead *.c set showmatch
au BufNewFile,BufRead *.cc set showmatch

What if an unbalanced bracket enters your code, or worse, you need to catch unbalanced brackets in someone else's spaghetti code? The previously mentioned % editor command searches for balanced grouping characters. For example, if you place the cursor over a left square bracket, [, and type % from command mode, Vim will reposition the cursor to the next ] character. If you place the cursor on a right curly brace, }, and invoke %, then Vim repositions the cursor on the previous matching { character. In this way, you can verify not only that any given round, curly, or square bracket has a corresponding matching partner, but also that the partner matches semantically. You can even define other matching pairs of "brackets," like the HTML comment delimiters <!-- and -->, using Vim's matchpair command. See the Vim help pages for more information.

Invoking make from within Vim can be very handy. For instance, instead of manually saving your file and typing make clean in another window, all you need to do is type :make clean from command mode. (Make sure autowrite is set, so that Vim automatically saves the file before running the make command.) In general, whenever you type

:make arguments

from command mode, Vim will run make and pass arguments to it.

It gets even better. When you make your program from within Vim, the editor captures all the messages that the compiler issues. It understands the syntax of GCC's output and knows when a compiler warning or error occurs. Let's take a look at this in action. Consider the following code:


It looks like somebody's been doing a little concurrent Fortran and C coding! Suppose you're currently editing main.c and want to build the program. Issue the :make command from within Vim and see all the error messages (Figure 7-4).

Now if you press ENTER or the spacebar, you return to editing the program, but with the cursor positioned on the line that generated the first warning or error (in this case, the message that argc was undeclared) as shown in Figure 7-5.



After you fix the error, there are two ways to proceed to the next error:

Becoming proficient with your chosen editor is so self-evident that it's often neglected, but it is really the first step in learning to program in a particular environment. At the risk of overstating things, editors are to programmers as musical instruments are to musicians. Even the most creative composers need to know the basics of how to play an instrument in order to realize their ideas so that other people can benefit from them. Learning to use your editor to its fullest extent enables you to program more quickly, figure out other people's code more effectively, and reduce the number of compile cycles you need to perform when debugging code.

If you use Vim, we recommend Steve Oualline's Vi IMproved—Vim (New Riders, 2001). The book is thorough and well written. (Unfortunately, it was written for Vim 6.0, and Vim 7.0 and later features like folding are not covered.) Our purpose here was just to give a taste of the things that Vim can do for the programmer, but Steve's book is a great resource for learning the details.

Vim has many features that the authors find obscenely useful. For example, we would have liked to cover

But this is a book on debugging, not on Vim, and we need to get back to discussing additional software tools.



[16] This figure needed to be converted to grayscale for the book. In practice, identifying the error would be even easier.

[17] Admit it. You were expecting them not to balance. Our point is not that you were wrong, but that it probably took you longer than it should have to answer the question!