Debugga C kod med emacs och gdb

From Peters wiki
Jump to navigation Jump to search

Denna guide visar hur man kommer igång med att debugga C kod med gdb. gdb startas i emacs för att kunna på ett enklare sätt sätta brytpunkter, titta på variabler, titta på koden m.m. Program som måste vara installerade:

  • gcc
  • gdb
  • emacs

Ett första exempel

Följande kod används i denna guide:

main.c
#include <stdio.h>

int main(void)
{
  int i;
  int sum;

  sum = 0;
  for (i = 0; i < 100; i++) sum += i;
  printf("sum = %d\n", sum);
	
  return 0;
}

Kompilera koden med debuggstöd

Kompilera koden med debuggstöd:

user $ gcc -o calcsum -g main.c

Start gdb i emacs

Börja med att starta igång emacs i samma katalog som koden och ditt kompilerade program:

user $ emacs &

Starta sedan gdb i emacs genom att trycka på M+x (Meta fås genom att trycka på Alt-knappen) och skriva kommandot gdb. Emacs kommer sedan att fråga:

Run gdb (like this): gdb --annotate=3 calcsum

Tryck bara på Enter för att acceptera kommandot. Börja sedan med att sätta en brytpunkt på funktionen main():

user $ (gdb) break main
Breakpoint 1 at 0x400480: file main.c, line 8.

Kör sedan programmet fram till denna brytpunkt genom att köra kommandot run eller r:

user $ (gdb) r
Starting program: /home/peter/tmp/gdb/calcsum
Breakpoint 1, main () at main.c:8

Ett fönster med koden ska nu öppnas i ett emacs fönster:


Öppna ett fönster i emacs som visar alla lokala variabler i nuvarande funktion genom menyn Gud => GDB-Windows => Locals. Exekvera sedan rader rader genom att köra kommandot next eller n. Antingen kan man köra kommadot next två gånger eller så anger man hur många rader man vill köra efter kommandot:

user $ (gdb) n 2

Man kan nu se i locals fönstret att i och sum har blivit uppdaterade till 100 resp. 4950:

Skillnaden mellan step och next

Anta att vi lägger till en funktion som sköter om utskriften av summan:

main.c
#include <stdio.h>

static void printsum(const int sum);

static void printsum(const int sum)
{
  printf("sum = %d\n", sum);
}

int main(void)
{
  int i;
  int sum;

  sum = 0;
  for (i = 0; i < 100; i++) sum += i;
  printsum(sum);
	
  return 0;
}

Om vi nu kompilerar om koden och laddar in den i gdb och kör next kommandot, så kommer vi märka att debuggern inte går in i funktionen printsum(). För att gå in i funktioner ska man använda kommandot step eller s i gdb. Starta om exekveringen med run eller r och prova med s istället när gdb kommer till printsum() raden. Nu kommer gdb hoppa in i funktionen printsum(). Om du har fönstret med lokala variabler öppet (locals) så kommer du också märka att det automatiskt uppdateras när du kliver in i printsum(). Variablerna i och sum försvinner, då dessa inte är synliga i funktionen printsum().

Databrytpunkt

Anta att vi är intresserade att stanna exekveringen när en variabel ändrar värde, men inte vet var i koden den ändras. Då är en databrytpunkt perfekt. Anta följande kod:

main.c
int a = 0;

void foo(void);

void foo(void)
{
  a++;
}

int main(void)
{
  int i;

  for (i = 0; i < 100; i++)
    if (i == 57) foo();
  
  return 0;
}

Kompilera koden och starta upp debuggningen i emacs. Gå till main() rutinen. Kör sedan kommandot:

user $ watch a

För att sätta en brytpunkt då a ändras. Fortsätt sedan exekveringen genom att köra kommandot:

user $ c

som är en förkortning för kommandot continue.

Exekveringen ska stanna i funktionen foo(), eftersom där räknas a upp med ett.