This chapter explains how to start, execute, save, restore, and quit a dbx debugging session. It contains the following sections:
How you start dbx depends on what you are debugging, where you are, what you need dbx to do, how familiar you are with dbx, and whether or not you have set up any dbx environment variables.
You can use dbx entirely from the command line in a terminal window, or run dbxtool, a graphical user interface for dbx. For information on dbxtool, see the dbxtool man page and the online help in dbxtool.
The simplest way to start a dbx session is to type the dbx command or dbxtool command at a shell prompt.
| $ dbx | 
or
| $ dbxtool | 
To start dbx from a shell and load a program to be debugged, type:
| $ dbx program_name | 
or
| $ dbxtool program_name | 
To start dbx and load a program that is a mixture of Java code and C JNI code or C++ JNI code:
| $ dbx program_name{.class | .jar} | 
Sun Studio software includes two dbx binaries, a 32-bit dbx that can debug 32-bit programs only, and a 64-bit dbx that can debug both 32-bit and 64-bit programs. When you start dbx, it determines which of its binaries to execute. On 64-bit operating systems, the 64-bit dbx is the default.
On the Linux OS, the 64-bit dbx cannot debug 32-bit programs. To debug a 32-bit program on the Linux OS, you must start the 32-bit dbx with the dbx command option -xexec32 or set the DBX_EXEC_32 environment variable.
When using the 32-bit dbx on a 64-bit Linux OS, do not use the debug command or set the follow_fork_mode environment variable to child if the result will be execution of a 64-bit program. Exit dbx and start the 64-bit dbx to debug a 64-bit program.
For more information on the dbx command and start-up options, see dbx Command and the dbx(1) man page.
If the program that dumped core was dynamically linked with any shared libraries, it is best to debug the core file in the same operating environment in which it was created. dbx has limited support for the debugging of “mismatched” core files (for example, core files produced on a system running a different version or patch level of the Solaris Operating System.
dbx cannot tell you the state of a Java application from a core file as it can with native code.
To debug a core file, type:
| $ dbx program_name core | 
or
| $ dbxtool program_name core | 
If you type the following, dbx determines the program_name from the core file:
| $ dbx - core | 
or
| $ dbxtool - core | 
You can also debug a core file using the debug command when dbx is already running:
| (dbx) debug -c core program_name | 
You can substitute - for the program name and dbx will attempt to extract the program name from the core file. dbx might not find the executable if its full path name is not available in the core file. If dbx does not find the executable, specify the complete path name of the binary when you tell dbx to load the core file.
If the core file is not in the current directory, you can specify its path name (for example, /tmp/core).
Use the where command (see where Command) to determine where the program was executing when it dumped core.
When you debug a core file, you can also evaluate variables and expressions to see the values they had at the time the program crashed, but you cannot evaluate expressions that make function calls. You cannot single step. You can set breakpoints and then rerun the program.
If you have problems loading a core file, check whether you have a truncated core file. If you have the maximum allowable size of core files set too low when the core file is created, then dbx cannot read the resulting truncated core file. In the C shell, you can set the maximum allowable core file size using the limit command (see the limit(1) man page). In the Bourne shell and Korn shell, use the ulimit command (see the limit(1) man page). You can change the limit on core file size in your shell start-up file, re-source the start-up file, and then rerun the program that produced the core file to produce a complete core file.
If the core file is incomplete, and the stack segment is missing, then stack trace information is not available. If the runtime linker information is missing, then the list of loadobjects is not available. In this case, you get an error message about librtld_db.so not being initialized. If the list of LWPs is missing, then no thread information, lwp information, or stack trace information is available.If you run the where command, you get an error saying the program was not “active.”
Sometimes a core file is created on one system (the core-host) and you want to load the core file on another machine (the dbx-host) to debug it. However, two problems with libraries might arise when you do so:
The shared libraries used by the program on the core-host might not be the same libraries as those on the dbx-host. To get proper stack traces involving the libraries, you’ll want to make these original libraries available on the dbx-host.
dbx uses system libraries in /usr/lib to help understand the implementation details of the run time linker and threads library on the system. It may also be necessary to provide these system libraries from the core-host so that dbx can understand the runtime linker data structures and the threads data structures.
The user libraries and system libraries can change in patches as well as major Solaris operating environment upgrades, so this problem can even occur on the same host, if, for example, a patch was installed after the core file was collected, but before running dbx on the core file.
dbx might display one or more of the following error messages when you load a “mismatched” core file:
| dbx: core file read error: address 0xff3dd1bc not available dbx: warning: could not initialize librtld_db.so.1 -- trying libDP_rtld_db.so dbx: cannot get thread info for 1 -- generic libthread_db.so error dbx: attempt to fetch registers failed - stack corrupted dbx: read of registers from (0xff363430) failed -- debugger service failed | 
 To Eliminate the Library Problems and Debug a “mismatched”
Core File
To Eliminate the Library Problems and Debug a “mismatched”
Core FileSet the dbx environment variable core_lo_pathmap to on.
Use the pathmap command to tell dbx where the correct libraries for the core file are located.
Use the debug command to load the program and the core file.
For example, assuming that the root partition of the core-host has been exported over NFS and can be accessed using /net/core-host/ on the dbx-host machine, you would use the following commands to load the program prog and the core file prog.core for debugging:
| (dbx) dbxenv core_lo_pathmap on (dbx) pathmap /usr /net/core-host/usr (dbx) pathmap /appstuff /net/core-host/appstuff (dbx) debug prog prog.core | 
If you are not exporting the root partition of the core-host, you must copy the libraries by hand. You need not re-create the symbolic links. (For example, you need not make a link from libc.so to libc.so.1; just make sure libc.so.1 is available.)
Keep the following things in mind when debugging a mismatched core file:
The pathmap command does not recognize a pathmap for ’/’ so you cannot use the following command:
pathmap / /net/core-host
The single-argument mode for the pathmap command does not work with load object pathnames, so use the two argument from-path to-path mode.
Debugging the core file is likely to work better if the dbx-host has either the same or a more recent version of the Solaris operating environment than the core-host, though this is not always necessary.
The system libraries that you might need are:
For the runtime linker:
/usr/lib/ld.so.1
/usr/lib/librtld_db.so.1
/usr/lib/64/ld.so.1
/usr/lib/64/librtld_db.so.1
For the threads library, depending on which implementation of libthread you are using:
/usr/lib/libthread_db.so.1
/usr/lib/64/libthread_db.so.1
You will need the 64-bit versions of the xxx_db.so libraries if dbx is running on a 64-bit capable version of the Solaris OS since these system libraries are loaded and used as part of dbx, not as part of the target program.
The ld.so.1 libraries are part of the core file image like libc.so or any other library, so you need the 32-bit ld.so.1 library or 64-bit ld.so.1 library that matches the program that created the core file.
If you are looking at a core file from a threaded program, and the where command does not display a stack, try using lwp commands. For example:.
| (dbx) where current thread: t@0 [1] 0x0(), at 0xffffffff (dbx) lwps o>l@1 signal SIGSEGV in _sigfillset() (dbx) lwp l@1 (dbx) where =>[1] _sigfillset(), line 2 in "lo.c" [2] _liblwp_init(0xff36291c, 0xff2f9740, ... [3] _init(0x0, 0xff3e2658, 0x1, ... ... | 
The -setfp and -resetfp options of the lwp command are useful when the frame pointer (fp) of the LWP is corrupted. These options work when debugging a core file, where assign $fp=... is unavailable.
The lack of a thread stack can indicate a problem with thread_db.so.1 Therefore, you might also want to try copying the proper libthread_db.so.1 library from the core-host.
You can attach a running process to dbx using the process ID as an argument to the dbx command or the dbxtool command.
| $ dbx program_name process_id | 
or
| dbxtool program_name process_id | 
To attach dbx to a running process that includes JavaTM code and C JNI (Java Native Interface) code or C++ JNI code:
| $ dbx program_name{.class | .jar} process_id
 | 
You can also attach to a process using its process ID without knowing the name of the program.
| $ dbx - process_id | 
or
| $ dbxtool - process_id | 
Because the program name remains unknown to dbx, you cannot pass arguments to the process in a run command.
For more information, see Attaching dbx to a Running Process.
When you start dbx,if you do not specify the -S option, dbx looks for the installed startup file, dbxrc, in the directory /installation_directory/lib. (The default installation_directory is /opt/sunstudio12.1 on Solaris platforms and /opt/sun/sunstudio12.1 on Linux platforms.) If your Sun Studio software is not installed in the default directory, dbx derives the path to the dbxrc file from the path to the dbx executable.
Then dbx searches for a .dbxrc file in the current directory, then in $HOME. You can specify a different startup file than .dbxrc explicitly by specifying the file path using the -s option. For more information, see Using the dbx Initialization File.
A startup file may contain any dbx command, and commonly contains the alias command, dbxenv command, pathmap command, and Korn shell function definitions. However, certain commands require that a program has been loaded or a process has been attached to. All startup files are loaded before the program or process is loaded. The startup file may also source other files using the source or .(period) command. You can also use the startup file to set other dbx options.
As dbx loads program information, it prints a series of messages, such as Reading filename.
Once the program is finished loading, dbx is in a ready state, visiting the “main” block of the program (for C or C++: main(); for Fortran 95: MAIN()). Typically, you set a breakpoint (for example, stop in main) and then issue a run command for a C program.
You can use the pathmap command, dbxenv command, and alias command to set startup properties for your dbx sessions.
By default, dbx looks in the directory in which the program was compiled for the source files associated with the program being debugged. If the source or object files are not there or the machine you are using does not use the same path name, you must inform dbx of their location.
If you move the source or object files, you can add their new location to the search path. The pathmap command creates a mapping from your current view of the file system to the name in the executable image. The mapping is applied to source paths and object file paths.
Add common pathmaps to your .dbxrc file.
To establish a new mapping from the directory from to the directory to, type:
| (dbx) pathmap [ -c ] from to | 
If -c is used, the mapping is applied to the current working directory as well.
The pathmap command is useful for dealing with automounted and explicit NFS-mounted file systems with different base paths on differing hosts. Use -c when you try to correct problems due to the automounter because current working directories are inaccurate on automounted file systems.
The mapping of /tmp_mnt to / exists by default.
For more information, see pathmap Command.
You can use the dbxenv command to either list or set dbx customization variables. You can place dbxenv commands in your .dbxrc file. To list variables, type:
| $ dbxenv | 
You can also set dbx environment variables. See Saving and Restoring Using replay for more information about the .dbxrc file and about setting these variables.
For more information, see Setting dbx Environment Variables and dbxenv Command.
You can create your own dbx commands using the kalias or dalias commands. For more information, see dalias Command.
You must prepare your program for debugging with dbx by compiling it with the- g or -g0 option.
The -g option instructs the compiler to generate debugging information during compilation.
For example, to compile using C++, type:
| % CC -g example_source.cc | 
In C++, the -g option turns on debugging and turns off inlining of functions. The- g0 (zero) option turns on debugging and does not affect inlining of functions. You cannot debug inline functions with the -g0 option. The -g0 option can significantly decrease link time and dbx start-up time (depending on the use of inlined functions by the program).
To compile optimized code for use with dbx, compile the source code with both the -O (uppercase letter O) and the -g options.
dbx lets you use options in the objcopy command on Linux platforms and the gobjcopy command on Solaris platforms to copy the debugging information from an executable to a separate debug file, strip that information from the executable, and create a link between these two files.
dbx searches for the separate debug file in the following order and reads the debugging information from the first file it finds:
The directory that contains the executable file.
A subdirectory named debug in the directory that contains the executable file.
A subdirectory of the global debug file directory, which you can view or change if the dbx environment variable debug_file_directory is set to the path name of the directory. The default value of the environment variable is /usr/lib/debug.
For example, to create a separate debug file for executable a.out, you would do the following.
 Creating a Separate Debug File
Creating a Separate Debug FileCreate a separate debug file named a.out.debug containing the debugging information.
Strip the debugging information from a.out.
Establish the link between the two files. On Solaris platforms, use the gobjcopy command. On Linux platforms, use the objcopy command.
On a Linux platform, you can use the command objcopy -help to find out whether or not the -add-gnu-debuglink option is supported on the platform. You can replace the -only-keep-debug option of the objcopy command with the command cp a.out a.out.debug to make a.out.debug a fully executable file.
dbx provides partial debugging support for optimized code. The extent of the support depends largely upon how you compiled the program.
When analyzing optimized code, you can:
Stop execution at the start of any function (stop in function command)
Evaluate, display, or modify arguments
Evaluate, display, or modify global, local, or static variables
Single-step from one line to another (next or step command)
When programs are compiled with optimization and debugging enabled at the same time (using the -O -g options), dbx operates in a restricted mode.
The details about which compilers emit which kind of symbolic information under what circumstances is considered an unstable interface and is likely to change from release to release.
Source line information is available, but the code for one source line might appear in several different places for an optimized program, so stepping through a program by source line results in the “current line” jumping around in the source file, depending on how the code was scheduled by the optimizer.
Tail call optimization can result in missing stack frames when the last effective operation in a function is a call to another function.
Generally, symbolic information for parameters, local variables, and global variables is available for optimized programs. Type information about structs, unions, C++ classes, and the types and names of local variables, global variables, and parameters should be available. The C++ compiler does not provide symbolic type information about local variables; the C compiler does.
Information about the location of parameters and local variables is sometimes missing for optimized code. If dbx cannot locate a value, it reports that it cannot. Sometimes the value might disappear temporarily, so try to single-step and print again. If you stop at the first instruction of an optimized function, dbx can print the values of parameters, because the values will be in ABI-conforming registers or stack locations.
The Sun Studio 12 Update 1 compilers for SPARC based systems provide the information for locating parameters and local variables. Newer versions of the GNU compilers also provide this information. The Sun Studio 12 Update 1 compilers for x86 based systems, and compilers in the Sun Studio 12 release and older Sun Studio releases do not provide this information.
You can print global variables and assign values to them, although they might have inaccurate values if the final register-to-memory store has not happened yet.
The sections Inlining of Functions and Parallelization in Chapter 8 of the Sun Studio Performance Analyzer manual contain information that might be helpful when debugging an optimized program.
For OpenMP programs, compiling with the -xopenmp=noopt option instructs the compiler not to apply any optimizations. However, the optimizer still processes the code in order to implement the OpenMP directives, so some of the problems described might occur in programs compiled with -xopenmp=noopt.
While most debugging support requires that a program be compiled with -g, dbx still provides the following level of support for code compiled without -g:
Backtrace (dbx where command)
Calling a function (but without parameter checking)
Checking global variables
Note, however, that dbx cannot display source code unless the code was compiled with the -g option. This restriction also applies to code that has had strip -x applied to it.
For full support, a shared library must also be compiled with the -g option. If you build a program with shared library modules that were not compiled with the -g option, you can still debug the program. However, full dbx support is not possible because the information was not generated for those library modules.
dbx can debug programs that have been completely stripped. These programs contain some information that can be used to debug your program, but only externally visible functions are available. Some runtime checking works on stripped programs or load objects: memory use checking works, and access checking works with code stripped with strip -x but not with code stripped with strip.
A dbx session runs from the time you start dbx until you quit dbx; you can debug any number of programs in succession during a dbx session.
To quit a dbx session, type quit at the dbx prompt.
| (dbx) quit | 
When you start dbx and attach it to a running process using the process_id option, the process survives and continues when you quit the debugging session. dbx performs an implicit detach before quitting the session.
You can stop execution of a process at any time by pressing Ctrl+C without leaving dbx.
If you have attached dbx to a process, you can detach the process from dbx without killing it or the dbx session by using the detach command.
To detach a process from dbx without killing the process, type:
| (dbx) detach | 
You can detach a process and leave it in a stopped state while you temporarily apply other /proc-based debugging tools that might be blocked when dbx has exclusive access. For more information, see Detaching dbx From a Process.
For more information on the detach command, see detach Command.
The dbx kill command terminates debugging of the current process as well as killing the process. However, the kill command preserves the dbx session itself leaving dbx ready to debug another program.
Killing a program is a good way of eliminating the remains of a program you were debugging without exiting dbx.
To kill a program executing in dbx, type:
| (dbx) kill | 
For more information, see kill Command.
dbx provides three commands for saving all or part of a debugging run and replaying it later:
The save command saves to a file all debugging commands issued from the last run command, rerun command, or debug command up to the save command. This segment of a debugging session is called a debugging run.
The save command saves more than the list of debugging commands issued. It saves debugging information associated with the state of the program at the start of the run: breakpoints, display lists, and the like. When you restore a saved run, dbx uses the information in the save-file.
You can save part of a debugging run; that is, the whole run minus a specified number of commands from the last one entered.
| debug | debug | ||
| stop at line | stop at line | ||
| run | run | ||
| next | next | ||
| next | next | ||
| Saving a complete run | stop at line | Saving a run minus | stop at line | 
| continue | the last two steps | continue | |
| next | next | ||
| next | next | ||
| step | step | ||
| next | next | ||
| save | save -2 | ||
If you are not sure where you want to end the run you are saving, use the history command to see a list of the debugging commands issued since the beginning of the session.
By default, the save command writes information to a special save-file. If you want to save a debugging run to a file you can restore later, you can specify a file name with the save command. See Saving a Series of Debugging Runs as Checkpoints .
To save an entire debugging run up to the save command, type:
| (dbx) save | 
To save part of a debugging run, use the save number command, where number is the number of commands back from the save command that you do not want saved.
| (dbx) save -number | 
If you save a debugging run without specifying a file name, dbx writes the information to a special save-file. Each time you save, dbx overwrites this save-file. However, by giving the save command a filename argument, you can save a debugging run to a file that you can restore later, even if you have saved other debugging runs since the one saved to filename.
Saving a series of runs gives you a set of checkpoints, each one starting farther back in the session. You can restore any one of these saved runs, continue, then reset dbx to the program location and state saved in an earlier run.
To save a debugging run to a file other than the default save-file:
| (dbx) save filename | 
After saving a run, you can restore the run using the restore command. dbx uses the information in the save-file. When you restore a run, dbx first resets the internal state to what it was at the start of the run, then reissues each of the debugging commands in the saved run.
The source command also reissues a set of commands stored in a file, but it does not reset the state of dbx; it only reissues the list of commands from the current program location.
For exact restoration of a saved debugging run, all the inputs to the run must be exactly the same: arguments to a run-type command, manual inputs, and file inputs.
If you save a segment and then issue a run, rerun, or debug command before you do a restore, restore uses the arguments to the second, post-save run, rerun, or debug command. If those arguments are different, you might not get an exact restoration.
To restore a saved debugging run, type:
| (dbx) restore | 
To restore a debugging run saved to a file other than the default save-file, type:
| (dbx) restore filename | 
The replay command is a combination command, equivalent to issuing a save -1 followed immediately by a restore. The replay command takes a negative number argument, which it passes to the save portion of the command. By default, the value of -number is -1, so the replay command works as an undo command, restoring the last run up until, but not including, the last command issued.
To replay the current debugging run, minus the last debugging command issued, type:
| (dbx) replay | 
To replay the current debugging run and stop the run before a specific command, use the dbx replay command, where number is the number of commands back from the last debugging command.
| (dbx) replay -number |