Valgrind

Linux.fista
Versio hetkellä 6. helmikuuta 2009 kello 13.48 – tehnyt Jarkko (keskustelu | muokkaukset) (helgrind, wikitystä)
Siirry navigaatioon Siirry hakuun
valgrind
Käyttöliittymä teksti
Lisenssi GPL
Kotisivu valgrind.org

Valgrind on helppokäyttöinen ja hyödyllinen muistidebuggaus- ja profilointityökalu Linuxille, joka tukee x86-, AMD64-, ppc- ja ppc64-prosessorityyppejä.

Käyttö

Valgrind käynnistetään kuten Gdb

valgrind ./ohjelma

Tämän jälkeen Valgrind tulostaa, oletuksena standarditulostusvirtaan, mm. hyödyllistä tietoa ohjelman muistinkäytön puutteista, jos niitä on.

Työkalut

Mikäli haluat käyttää jotain muuta työkalua kuin memcheck, käytä valitsinta --tool=<työkalu>.

Valittavana on seuraavat työkalut:

  • Memcheck, erinomainen työkalu muistinhallinnan virheiden löytämiseen.
  • Cachegrind, työkalu, joka kertoo kuinka hyvin ohjelmasi hyödyntää prosessorivälimuistia.
  • Callgrind, työkalu, joka antaa kehittäjälle olennaista tietoa ohjelman eri osien käyttämistä prosessoriresursseista .
  • Massif, työkalu ohjelman dynaamisesti varattavan muistin käytön profilointiin.
  • Helgrind, synkronointivirheiden etsimiseen POSIX-säikeitä käyttävistä ohjelmista.

Tulosteen tulkinta

Valgrindin tulostamien rivien alussaoleva ==PID== kertoo senhetkisen ajettavan prosessin prosessitunnuksen. Ohjelman oma tuloste taas tulostuu sellaisenaan ilman tätä merkintää.

tool=memcheck

==12799== Invalid read of size 4

Edellinen merkintä kertoo, että muistista on luettu alue standardikirjastolta varattujen muistialueiden ulkopuolelta. "of size 4" kertoo tässä, että luvun koko on neljän tavun eli 32 bitin kokoinen. Tämä on ehdottomasti virhe, tarkista esimerkiksi taulukoiden indeksoinnit. Merkinnän jälkeen tulostuu kutsupino, joka kertoo, missä virhe tapahtuu. Esimerkiksi:

==12393==    at 0x80483FA: main (testi.c:7)

kertoo, että virhe tapahtuu suoritettavan ohjelman kohdassa 0x80483FA, funktiossa main(), lähdekooditiedostossa testi.c ja rivillä 7.

==16340==  Address 0x416702D is 1 bytes after a block of size 4 alloc'd

Kertoo luetun muistiosoitteen ja hyödyllistä tietoa sen sijainnista suhteessa varattuun muistiin.

==17298== Invalid write of size 1

Merkintä kertoo, että muistiin on kirjoitettu alue varattujen muistialueiden ulkopuolelta. Tulkinta on lähes täysin vastaava edelliseen nähden.

==18096== Conditional jump or move depends on uninitialised value(s)

Tämä virhe syntyy, kun koodissa on käytetty if-, for- tai while-ohjauslausekkeiden ehtona muuttujaa, jonka muistialue on alustamaton tai jonka muistialueen sisältö ei ole määräytynyt täsmällisesti ohjelman ajon aikana.

Tässä vikana voi olla esimerkiksi unohtunut = NULL -sijoitus olemattomalle oliolle, kun olion ja NULL:n vertailua käytetään ehtona olion käsittelylle. Kannattaa kuitenkin välttää logiikkavirheiden peittämistä mielivaltaisella alustamisella!

Yhteenveto ohjelman muistinkäytöstä

==8843== ERROR SUMMARY: 4 errors from 4 contexts (suppressed: 7 from 1)
==8843== malloc/free: in use at exit: 4 bytes in 1 blocks.
==8843== malloc/free: 1 allocs, 0 frees, 4 bytes allocated.
==8843== For counts of detected errors, rerun with: -v
==8843== searching for pointers to 1 not-freed blocks.
==8843== checked 53,100 bytes.

Käyttämällä valitsinta --leak-check=full saadaan tietoa menetetystä muistista:

==8843== 4 bytes in 1 blocks are definitely lost in loss record 1 of 1
==8843==    at 0x4021628: malloc (in /usr/lib/valgrind/x86-linux/vgpreload_memcheck.so)
==8843==    by 0x80483F1: main (testi.c:7)

Tämä kertoo, että kadotettu 4 tavun kokoinen 1 muistialue (block) varattiin käyttöön testi.c:n rivillä 7. Monimutkaisemmissa tilanteissa tulostuu koko kutsupino, joka aiheutti muistinvaraamisen. Se on kuitenkin tulkittavissa, kuten aiemmin esitettiin.

==8843== LEAK SUMMARY:
==8843==    definitely lost: 4 bytes in 1 blocks.
==8843==      possibly lost: 0 bytes in 0 blocks.
==8843==    still reachable: 0 bytes in 0 blocks.
==8843==         suppressed: 0 bytes in 0 blocks.
==8843== Reachable blocks (those to which a pointer was found) are not shown.
==8843== To see them, rerun with: --show-reachable=yes

Tämä kertoo yhteenvedon hukatusta muistista - hyvässä C-kielisessä ohjelmassa tyypillisesti kaikki luvut tässä ovat nollia.

"definitely lost" kertoo, että muistialueen käsittely voidaan varmasti tulkita muistivuodoksi.

"possibly lost" kertoo, että muistialueen käsittely voidaan luultavasti tulkita muistivuodoksi.

"still reachable" kertoo, että varattuun muistialueeseen on säilytetty viittaus ohjelman loppuun asti, mutta sitä ei vapauteta ennen ohjelma päättymistä. Tunnetusti C++:n standardikirjastojen templaattisäilöt eivät vapauta muistia välttämättä käskettäessäkään, vaan säilyttävät varauksen, jotta alue voitaisiin ottaa tarvittaessa uudelleen käyttöön. Tästä ei siis tarvitse välttämättä C++-ohjelmissa huolestua.

Vaihtoehtoiset käyttöliittymät

Aiheesta muualla

Katso myös