Профилирование кода на C/C++ в Linux и FreeBSD

22 марта 2016

Рассмотрим типичную ситуацию — приходит пользователь и говорит «я делаю в приложении то-то и то-то и у меня все сильно тормозит». Как понять, в чем проблема, и исправить ее? Вот об этом мы и поговорим в данной заметке. Приведенный список методов не претендует на полноту, однако это то, что в настоящее время я чаще всего использую на практике. Несмотря на то, что пишу я сейчас в основном на C, описанные методы будут также применимы в проектах, использующих С++, и, возможно, другие языки программирования.

Дедовский метод

Начнем с самого примитивного, но, тем не менее, весьма популярного способа. Идея простая — замеряем время в начале выполнения операции, замеряем в конце, считаем дельту, выводим в лог. Вот так это примерно выглядит с использованием процедуры gettimeofday:

#include <sys/time.h>

/* ... */

struct timeval  tv;
gettimeofday(&tv, NULL);

double time_begin = ((double)tv.tv_sec) * 1000 +
                      ((double)tv.tv_usec) / 1000;

/* ... */

gettimeofday(&tv, NULL);
double time_end = ((double)tv.tv_sec) * 1000 +
                    ((double)tv.tv_usec) / 1000 ;

double total_time_ms = time_end - time_begin;

printf("TOTAL TIME (ms) = %f\n", total_time_ms);

Примечание: См также кроссплатформенную реализацию процедуры getCurrentTimeMs из заметки Продолжаем изучение OpenGL: простой вывод текста.

Пропарсить полученный лог можно, к примеру, с использованием Perl:

cat test.log | perl -e '$max = 0; while(<STDIN>) { $_ =~ /TOTAL TIME'\
' \(ms\) = (.*?)\,/; if($1 > 1.5) { print "$_"; } $max = $1 if $1 > '\
'$max; } print "max = $max\n"; '

Несмотря на тупость метода, он очень удобен, например, когда в приложении что-то тормозит иногда, а не постоянно. При использовании метода нужно помнить, что системный вызов gettimeofday сам по себе сравнительно дорогой. Если использовать его, скажем, между захватом и освобождения спинлока, бутылочное горлышко, которое вы ищите, может уехать вообще в другое место. Также вы можете получать неправильные цифры, если между двумя замерами времени шедулер решит переключиться на выполнение другого процесса.

Утилиты strace и ltrace

Программа strace показывает системные вызовы и их возвраты, что бывает довольно удобно при отладке. С флагом -c она показывает топ системных вызовов, их количество, и сколько времени было проведено в вызове:

$ strace -c pwd
/home/ubuntu
% time     seconds  usecs/call     calls    errors syscall
------ ----------- ----------- --------- --------- ----------------
 31.73    0.000132          15         9           mmap
 19.95    0.000083          28         3         3 access
 11.30    0.000047          16         3           open
 10.58    0.000044          22         2           munmap
  8.17    0.000034           9         4           mprotect
  4.57    0.000019           5         4           fstat
  3.37    0.000014           3         5           close
  2.88    0.000012          12         1           write
  2.64    0.000011           4         3           brk
  1.68    0.000007           7         1           execve
  1.44    0.000006           6         1           read
  0.96    0.000004           4         1           getcwd
  0.72    0.000003           3         1           arch_prctl
------ ----------- ----------- --------- --------- ----------------
100.00    0.000416                    38         3 total

Программа ltrace делает то же самое и для библиотечных вызовов:

$ ltrace -c pwd
/home/ubuntu
% time     seconds  usecs/call     calls      function
------ ----------- ----------- --------- --------------------
 17.31    0.000636         636         1 setlocale
 12.93    0.000475         118         4 __freading
  9.71    0.000357         178         2 fclose
  8.11    0.000298         298         1 puts
  6.56    0.000241         120         2 fileno
  5.69    0.000209         209         1 getenv
  5.61    0.000206         103         2 fflush
  5.52    0.000203         203         1 strrchr
  5.36    0.000197          98         2 __fpending
  5.25    0.000193         193         1 __cxa_atexit
  4.24    0.000156         156         1 bindtextdomain
  4.16    0.000153         153         1 textdomain
  3.86    0.000142         142         1 getcwd
  2.94    0.000108         108         1 getopt_long
  2.75    0.000101         101         1 free
------ ----------- ----------- --------- --------------------
100.00    0.003675                    22 total

Чтобы подключиться при помощи strace или ltrace к уже работающему процессу, используйте флаг -p.

Во FreeBSD утилита, аналогичная strace, называется truss. Кроме того, strace и ltrace доступны в портах.

Использование gprof

Утилита gprof доступна как в Linux, так и во FreeBSD. Для примера попробуем прогнать ее на программе из заметки Реализация хэш-таблиц, почти как в Perl.

Если программа уже была собрана, делаем make clean. Затем пересобираем ее с флагом -pg:

gcc -pg -Wall htable.c htable_test.c -o htable_test

Запускаем программу без gprof:

cat test.txt | ./htable_test > /dev/null

Будет создан двоичный файл gmon.out. Этот шаг обязателен, иначе при попытке профайлинга увидим ошибку:

gmon.out: No such file or directory

Теперь запускаем программу под gprof:

cat test.txt | gprof ./htable_test > profile

При этом сама программа в stdout ничего выводить не будет. Полученный текстовый файл profile вполне читаемый — видно, где и сколько времени проводила программа. Так, к примеру, могут выглядеть первые несколько его строк:

Flat profile:

Each sample counts as 0.01 seconds.
  %   cumulative   self              self     total          
 time   seconds   seconds    calls  ns/call  ns/call  name    
 30.37      0.10     0.10   475254   210.86   295.21  htable_del
 18.22      0.16     0.06  1425762    42.17    42.17  _htable_hash
 18.22      0.22     0.06   475254   126.52   210.86  htable_set
 12.15      0.26     0.04   950508    42.17    42.17  _htable_resize
 12.15      0.30     0.04   475254    84.34   126.52  htable_get
  3.04      0.31     0.01                             main
  0.00      0.31     0.00        1     0.00     0.00  htable_free
  0.00      0.31     0.00        1     0.00     0.00  htable_new

Также можно построить красивый граф вызовов:

sudo apt-get install graphviz
sudo pip install gprof2dot

gprof2dot ./profile | dot -Tsvg -o output.svg

Пример полученной картинки (кликабельно, SVG, ~ 9 Кб):

gprof2dot, пример результата

Подробности про утилиту dot и Graphviz см в заметке Рисуем красивые графы при помощи Graphviz.

Получение стэктрейсов при помощи Gdb

Как ни странно, gdb можно использовать и для профайлинга. Просто говорим:

gdb --batch --command=gdb.script -p 12345

… где в gdb.script написано:

bt

Понятно, что таким образом можно собирать не только стэктрейсы, но и другую информацию, например, значения переменных, переданных аргументов, и так далее. Этот прием показал себя весьма полезным, например, при поиске и устранении lock contention. Если программа часто висит в ожидании лока, мы будем видеть это ожидание по стэктрейсам.

Профайлинг при помощи perf top

Установка perf в Ubuntu / Debian:

sudo apt-get install linux-tools-common

При первом запуске увидим что-то вроде:

$ perf top -u postgres
WARNING: perf not found for kernel 3.13.0-71

You may need to install the following packages for this
specific kernel:
    linux-tools-3.13.0-71-generic
    linux-cloud-tools-3.13.0-71-generic

You may also want to install one of the following packages to keep
up to date:
    linux-tools-generic
    linux-cloud-tools-generic

Ставим и эти пакеты тоже:

sudo apt-get install linux-tools-generic linux-cloud-tools-generic

Можно посмотреть top по всей системе:

sudo perf top -a

… по процессам конкретного пользователя:

sudo perf top -u postgres

… или конкретному процессу:

sudo perf top -p 12345

Вот как это примерно выглядит:

Пример вывода perf top

Картинка обновляется в реальном времени. При помощи стрелочек и клавиши Enter можно «проваливаться внутрь» процессов и функций, вплоть до подсвечивания строчек кода и ассемблерных инструкций, которые тормозят. Очень удобно!

Во FreeBSD, насколько я понимаю, аналогичный функционал предоставляется программой pmcstat. Однако на практике я ею пока не пользовался.

Строим флемграфы

Помимо отображения топа самых часто вызываемых процедур программа perf умеет много чего еще.

Например, пишем stack samples с частотой 99 Герц для определенного pid’а со сборкой данных о call chains (флаг -g):

sudo perf record -p 12345 -F 99 -g

Запуск конкретной программы под perf производится так:

sudo perf record -F 99 -g -- ./myprog arg1 arg2 arg3

На выходе получаем файл perf.data.

Смотрим отчет:

sudo perf report --stdio

Но читать его в таком виде не очень-то удобно. Намного удобнее построить флеймграф:

git clone https://github.com/brendangregg/FlameGraph
sudo perf script | ./FlameGraph/stackcollapse-perf.pl > out.perf-folded
./FlameGraph/flamegraph.pl out.perf-folded > perf.svg

Получаем красивую картинку вроде такой:

Пример флеймграфа

В полученном SVG-файле можно кликать на названия процедур для увеличения соответствующего участка флеймграфа, а также искать процедуры по имени. Здесь и здесь можно найти больше информации и примеров. Еще этот набор скриптов умеет работать с dtrace и строить разные другие виды флеймграфов, но это уже тема для отдельной заметки.

Дополнение: Описание того, как строятся флеймграфы при помощи DTrace, вы найдете в заметке Использование DTrace на примере FreeBSD и Linux. Аналогичную инструкцию для SystemTap вы найдете в посте Установка и простые примеры использования SystemTap, а для bcc/eBPF — в статье Трассировка и профайлинг в Linux с помощью bcc/eBPF. Если же вас интересует профилирование памяти, читайте заметку Профилируем использование памяти в программах на C/C++ при помощи Heaptrack.

Заключение

Как видите, недостатка в инструментах профайлинга в мире C/C++ не наблюдается. Не один из описанных приемов не является универсальным, но при правильном их комбинировании можно разобраться практически в любой проблеме.

Метки: , , , .


Вы можете прислать свой комментарий мне на почту, или воспользоваться комментариями в Telegram-группе.