Similarly to many other interactive applications, one of the goals of a debugger is to be responsive to user requests, regardless of the internal state of the application.
This is not always easy to achieve without careful consideration during the design of the debugger.
Let's consider the different approaches that can be used.
Responsiveness requires that the debugger be ready to reply to external events with a meaningful response within a given amount of time. Current debuggers don't always realize this goal because they use one of the two following approaches:
  • they are single-threaded, thus using as much time as they need to compute a value, and relying on the operating system's signaling mechanism to interrupt the current operation (e.g. gdb). The main problem with this approach is that it's very difficult to keep the state of the debugger consistent when an interrupt is received from the user. One cannot know how many data structures are stable before interrupting the current operation and jumping to receive the next command. The use of global variables increases the difficulty of understanding and maintaining the code.
  • They use separate threads extensively, with different threads dedicated to different operations (such as reading registers, disassembling code, computing expressions etc.) The main problem with this approach is that it's difficult to coordinate the different threads when they need to interact among each other (for example an expression may need to read target memory at the same time as the disassembler needs to read the opcodes of the target program). Thread synchronization requires extensive use of semaphores and serialized access to shared resources such as symbol table and target communication.
A possible different approach is the very limited use of 2 separate threads to monitor the input/output channels (from/to the user interface and from/to the target). However, the main computational code of the debugger is mostly run by the main (default) thread and is kept sequential, operating on task objects that can switch between different states:

Ready A task is in this state when it has all information it needs to perform some computation. This could be because the command has just been received from the user, or because all needed data has been received from the target. The main thread of the debugger will switch one task from the ready state to the running state, and will start executing the associated computation.
Running Only one task is running at any given time. It stays in the running state until it either completes its computation, or it needs to wait for data that is not currently available. Only the main thread of the debugger executes running tasks. Because there is only one thread, there is no synchronization problem. The state of the debugger is guaranteed to be consistent whenever a task changes state.
Waiting A running task can enter the waiting state when it requires data, usually from the target. This could be reading a memory area or registers or waiting for the target to stop execution. When data is received by one of the I/O threads, the data is stored in the waiting task’s object and the state is switched to ready. The task will continue its computation when the main thread of the debugger will consider this task again. Note that a waiting task can also be interrupted by the I/O thread that is listening to the user’s input. In this case the I/O thread simply sets the state of the task to interrupted and leave to the main thread of the debugger to release the interrupted task.
Complete Once a running task has finished its computation, or it has been marked as interrupted, the main thread of the debugger will return the result to the user and destroy the task.

The following diagram shows the lifetime of a task:


© 2007 Giampiero Caprino, Backer Street Software