Gem #121 Breakpoint Commands — Part 1
by Jerome Guitton —AdaCore
Let's get started...
Consider this basic model of a Bank:
package Bank is procedure Open_Account (Cash : Integer); procedure Deposit (Cash : Integer); procedure Withdraw (Cash : Integer); end Bank;
This bank maintains only one account that has to be open before doing any transation. The only client of this bank would be a main subprogram P that opens an account and operates several transactions:
with Bank; procedure P is begin Bank.Open_Account (100); Bank.Deposit (45); Bank.Withdraw (55); Bank.Deposit (80); end P;
The implementation of package Bank is straightforward; it also contains an additional transaction at elaboration time, that corresponds to a special offer of the bank (and one may already see that it will cause a problem):
package body Bank is Balance : Integer; procedure Open_Account (Cash : Integer) is begin Balance := Cash; end Open_Account; procedure Deposit (Cash : Integer) is begin Balance := Balance + Cash; end Deposit; procedure Withdraw (Cash : Integer) is begin Balance := Balance - Cash; end Withdraw; -- To celebrate the inauguration of the bank at elaboration time, your -- account is credited with $100. Special_Offer : constant Integer := 100; begin Deposit (Special_Offer); end Bank;
There is a implicit property that the account should be opened only once, and before any transaction takes place; would this special offer break this property? This is a good opportunity to see how GDB can help us in checking properties of this form and break whenever the property is violated at run time. To do so, one may use the script language of GDB along with breakpoint commands.
The first way to do so would be set breakpoints on the bank operations:
(gdb) break open_account Breakpoint 1 at 0x401592: file bank.adb, line 7. (gdb) break deposit Breakpoint 2 at 0x4015a2: file bank.adb, line 12. (gdb) break withdraw Breakpoint 3 at 0x4015b8: file bank.adb, line 17.
However, we would not like to stop on any transaction, but only on an invalid one. At each breakpoint, we would like to check if our invariants hold, continue if they do, stop with an error if they don't.
First, the properties that we want to check require to maintain a state: whether or not the account has been open. This may be represented by a GDB convenience variable that can be created as follows:
set variable $account_open := False
We can then attach a small program that will be executed each time the breakpoint is hit. e.g. for breakpoint 1 on open_account:
(gdb) commands 1 >if $account_open = True >echo error: account opened twice >else >set variable $account_open := True >printf "(info) account created with $%d ", cash >continue >end >end
This command will resume the program if the account has not been opened yet, and will stop with an error message if we are trying to open it for the second time. The same kind of program may be used for withdraw and deposit:
(gdb) commands 2 >if $account_open = False >echo error: someone tries to deposit before opening an account >else >printf "(info) deposit: $%d ", cash >continue >end >end (gdb) commands 3 >if $account_open = False >echo error: someone tries to withdraw before opening an account >else >printf "(info) withdrawal: $%d ", cash >continue >end >end
Now let us run the program and look for invariant violations. One happens immediately:
(gdb) run
Breakpoint 2, bank.deposit (cash=100) at bank.adb:12 12 Balance := Balance + Cash; error: someone tries to deposit before opening an account (gdb)
Indeed, the special offer is credited before the account is opened:
(gdb) up #1 0x004015da in () at bank.adb:25 25 Deposit (Special_Offer);
Let us continue to find other violations. The program exits normally, meaning that no other violations were detected.
(gdb) c Continuing.
Breakpoint 1, bank.open_account (cash=100) at bank.adb:7 7 Balance := Cash; (info) account created with $100
Breakpoint 2, bank.deposit (cash=45) at bank.adb:12 12 Balance := Balance + Cash; (info) deposit: $45
Breakpoint 3, bank.withdraw (cash=55) at bank.adb:17 17 Balance := Balance - Cash; (info) withdrawal: $55
Breakpoint 2, bank.deposit (cash=80) at bank.adb:12 12 Balance := Balance + Cash; (info) deposit: $80 [Inferior 1 (process 7472) exited normally]
The script language of GDB provides various flow-control commands; you may find their full documentation in the section "Command files" in the GDB user's guide. We have seen that this language of GDB, along with convenience variables and breakpoint commands, provides a simple way to check some simple dynamic properties on a run. In order to check more elaborated properties, the full power of expression of the python interface should be used. This will be the subject of a forthcoming Gem.
About the Author
Jerome joined AdaCore in 2002 after completing his studies at the Ecole Nationale des Télécommunations (Paris, France), during which he had already worked with the company on one of its many reseach projects, namely PolyORB. His enthusiasm has remained undimmed during these six years and he has worked on a variety of projects, as well as acquiring expertise in debuggers and cross technologies. He has recently led the effort to port GNAT Pro to various cross targets (VxWorks 6, ELinOS).