Статическое и динамическое профилирование

22
Статическое и динамическое профилирование

Upload: ishmael-callahan

Post on 01-Jan-2016

62 views

Category:

Documents


0 download

DESCRIPTION

Статическое и динамическое профилирование. Исходные файлы. FE (C++/C или Fortran). Архитектура компилятора. Внутреннее представление. Профилировщик. Временный файл или Obj с ВП. Скалярные оптимизации. HPO. IP/IPO оптимизации. Генератор кода. Скалярные оптимизации. - PowerPoint PPT Presentation

TRANSCRIPT

Page 1: Статическое и динамическое профилирование

Статическое и динамическое профилирование

Page 2: Статическое и динамическое профилирование

FE (C++/C или Fortran)

Внутреннее представление

Профилировщик

Скалярные оптимизации HPO

Генератор кода

Исходные файлы

Обьектные файлы

Временный файл или Obj с ВП

IP/IPO оптимизации

Скалярные оптимизации

HPO Генератор кода

Исполняемый файлБиблиотека

Архитектура компилятора

Page 3: Статическое и динамическое профилирование

Определение выгодности оптимизаций Невыгодно сохранять общее подвыражение во временной переменной,

если «часто» используется только одно подвыражение.

z=x*y;

if(почти_никогда) {

t=x*y;

}

Невыгодно выносить инвариант из цикла, если он может ни разу не использоваться при расчетах.

for(i=0;i<n;i++) {

if(почти_никогда) {

t = invariant; break; }

}

Выгодно группировать вместе наиболее «часто» используемые фрагменты программы.

Невыгодно выполнять подстановку функции, которая «редко» вызывается в процессе выполнения программы.

Page 4: Статическое и динамическое профилирование

Выгодно комбинировать вместе «часто» совместно используемые элементы структуры.

Невыгодно тратить время на оптимизацию редко используемых фрагментов кода.

Статистический профилировщик вычисляет вероятность условных переходов и вес базовых блоков. Это делается на основе анализа исходного кода. Межпроцедурный анализ пытается рассчитать вес процедур на основе анализа графа вызовов.

Анализ исходного кода не может обеспечить точное вычисление весовых характеристик. В общем случае неизвестны входные данные исполняемой программы, время вычисления ограничено.

Существует встроенная функция builtin_expect, предназначенная для передачи компилятору информации о вероятности ветвления.

if(x) => if(__builtin_expect(x,1))

Page 5: Статическое и динамическое профилирование

Динамический профилировщик собирает весовые характеристики на основе анализа статистики, собранной при запуске инструментированного приложения.

Инструментация в данном случае – снабжение приложения механизмами для сбора статистики.

/Qprof-gen[:keyword]

instrument program for profiling.

Optional keyword may be srcpos or globdata

/Qprof-use[:<arg>]

enable use of profiling information during optimization

weighted - invokes profmerge with -weighted option to scale data

based on run durations

[no]merge - enable(default)/disable the invocation of the profmerge

tool

Page 6: Статическое и динамическое профилирование

Последовательность действий при использовании динамического профилировщика:

1. Построить приложение с инструментацией /Qprof_gen;

2. Запустить инструментированное приложение с представительным набором данных, т.е. одним или несколькими наборами наиболее часто используемых данных. Информация добавляется в файл со статистиками.

3. Пересобрать приложение с опцией /Qprof_use для использование собранных статистик при компиляции.

10/17/10

Page 7: Статическое и динамическое профилирование

Информация, собранная динамическим профилировщиком, более точная, поэтому некоторые оптимизации, применение которых может дать большой негативный эффект при неправильном применении, могут выполняться только при наличии информации от динамического профилировщика.

Некоторые оптимизации, которые базируются на профилировочной информации:

• перестановки базовых блоков;• группировка часто используемых функций;• вынос «холодных» базовых блоков за пределы

функции (расщепление функций).

10/17/10

Page 8: Статическое и динамическое профилирование

Динамическое выделение памяти Объекты и массивы могут инициализироваться динамически во время исполнения

приложения с использованием операторов new и delete, malloc и free. Менеджер памяти - это часть приложения, обрабатывающая запросы на выделение и освобождение памяти.

Типичные ситуация, когда динамическое выделение памяти необходимо:

• Необходимо создать большой массив, размер которого неизвестен во время компиляции.

• Массив может быть очень большим, для того чтобы расположить его на стеке.

• Объекты должны создаваться во время работы программы, но количество необходимых объектов неизвестно.

Неудобства динамического выделения памяти:

• Выделение и освобождение памяти имеет свою цену.

• Выделенная память становится фрагментированной, когда выделяются объекты различного типа в произвольном порядке. Это делает неэффективным кэширование данных.

• Если необходимо изменить размер созданного объекта, но нет возможности расширить блок, возникает потребность в копировании содержания старого блока памяти в новый блок.

• Необходима сборка мусора, поскольку в процессе работы могут исчезнуть блоки памяти необходимого размера.

Page 9: Статическое и динамическое профилирование

Важно оценить сильные и слабые стороны использования динамической памяти при проектировании приложения.

Различные менеджеры памяти реализуют различные алгоритмы выделения памяти и пытаются снизить затраты на работу с динамической памятью.

Альтернативные менеджеры памяти:

Smartheap

dlmalloc

Одним из важных факторов производительности в C++ является компактность размещения в памяти совместно используемых объектов, объединенных в различные связные списки. Связный список менее эффективен, чем линейный массив, по следующим причинам:

• Каждый объект создаётся отдельно. Выделение и освобождение объекта имеет свою цену.

• Объекты в памяти хранятся непоследовательно. При обходе связного списка вероятность попадания в кэш снижается. Процессорная предвыборка данных неэффективна.

• Необходима дополнительная память для хранения ссылок и информации о выделенных блоках памяти.

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

Page 10: Статическое и динамическое профилирование

Связные списки в памяти

10/17/10

Связный список:

Может располагаться в памяти так:

4GB

2GB

0GB

И в физической памяти:P1 P2

P3 P4

Page 11: Статическое и динамическое профилирование

#include <stdlib.h>#include <stdio.h>#define N 10000typedef struct { int x,y,z;} VecR;typedef VecR* VecP; int main() {int i,k;VecP a[N],b[N];VecR *tmp,*tmp1; #ifndef PERFfor(i=0;i<N;i++){ a[i]=(VecP)malloc(sizeof(VecR)); b[i]=(VecP)malloc(sizeof(VecR));}#elsetmp=(VecR*)malloc(sizeof(VecR)*N);tmp1=(VecR*)malloc(sizeof(VecR)*N);for(i=0;i<N;i++) { a[i]=(VecP)&tmp[i]; b[i]=(VecP)&tmp1[i];}#endif

for (i=0;i<N;i++){ a[i]->x = 1.0; b[i]->x = 2.0; a[i]->y = 2.0; b[i]->y = 3.0; a[i]->z = 0.0; b[i]->z = 4.0;} for(k=1;k<N;k++) { for (i=k;i<N-20;i++){ a[i]->x = b[i+10]->y+1.0; a[i]->y = b[i+10]->x+a[i+1]->y; a[i]->z = (a[i-1]->y - a[i-1]->x)/b[i+10]->y; }}printf("%d \n",a[100]->z);}

icc struct.c -fast -o a.outicc struct.c -fast -DPERF -o b.outtime ./a.out real 0m0.998stime ./b.out real 0m0.782s

Page 12: Статическое и динамическое профилирование

Контейнеры

Существует популярный способ улучшения работы с динамически выделяемой памятью при помощи контейнеров.

Создание и использование контейнеров - это один из примеров эффективного использования шаблонов (template) в С++. Наиболее распространенный набор контейнеров предоставляется Standard Template Library (STL), которая поставляется с современными C++ компиляторами.

Однако STL в основном разрабатывалась для гибкости использования, и вопросы производительности имели более низкий приоритет. Поэтому выделение памяти для хранения объектов осуществляется пошагово по мере роста количества объектов, которые необходимо хранить в памяти. Отсутствует гибкая система, позволяющая заранее определять, сколько памяти необходимо выделить изначально. В случае расширения контейнера может возникнуть необходимость копирования содержания контейнера. Это копирование происходит с использованием конструкторов копирования.

Еще один популярный метод – это метод пулов памяти. Одно из его отличий состоит в том, что при расширении пула весь блок памяти копируется с помощью memcpy.

Page 13: Статическое и динамическое профилирование

FE (C++/C или Fortran)

Внутреннее представление

Профилировщик

Скалярные оптимизации HPO

Генератор кода

Исходные файлы

Обьектные файлы

Временный файл или Obj с ВП

IP/IPO оптимизации

Скалярные оптимизации

HPO Генератор кода

Исполняемый файлБиблиотека

Архитектура компилятора

Page 14: Статическое и динамическое профилирование

Кодогенератор Кодогенерация — часть процесса компиляции, когда специальная часть

компилятора, кодогенератор, конвертирует синтаксически корректную программу в последовательность инструкций, которые могут выполнятся на машине. При этом могут применятся различные, в первую очередь машинно-зависимые оптимизации.

Часто кодогенератор является общей частью для множества компиляторов, каждый из которых генерирует промежуточный код, подаваемый на вход кодогенератору.

• Конвертация утверждений и выражений внутреннего представления в инструкции процессора данной архитектуры.

• Специфические архитектурные оптимизации. Удаление ветвления с помощью условных присваиваний.• Подставляет тела простейших встроенных функций.• Выравнивает базовые блоки в памяти.• Подготовка вызовов процедур, т.е. загрузка необходимых переменных в регистры

и/или на стек для передачи в вызываемую процедуру. • То же самое для вызываемой процедуры. Выделение места на стеке для

локальных переменных. Сохранение и восстановление используемых внутри процедуры регистров.

• Планирование инструкций. Планирование переходов. Учет задержки обращения к памяти. • Распределение регистров.• Вычисление дистанций для переходов.

Page 15: Статическое и динамическое профилирование

Распределение регистров

Одна из базовых задач кодогенератора – распределение регистров.

Распределением регистров называется отображение множества переменных программы на множество регистров микропроцессора.

Распределение регистров может выполняться в отдельно взятом базовом блоке (локальное распределение регистров) или во всей процедуре (глобальное распределение регистров).

Обычно количество переменных в программе значительно превосходит количество доступных физических регистров, поэтому некоторые переменные приходится откачивать в память. Стоимость откачки в память можно минимизировать, откачивая наименее часто используемые переменные, однако определить, какие переменные используются наименее часто, не всегда легко. Потеря производительности в связи с постоянным обменом между регистрами и памятью называется «вытеснением регистров» (register spilling).

Для распределения регистров используется метод раскраски графа несовместимости (register coloring).

Page 16: Статическое и динамическое профилирование

Раскраска

При реализации метода раскраски выполняются следующие шаги:• Определяются области жизни переменных (область программы, в

которой переменная используется), и каждой присваивается уникальное имя.

• Строится граф несовместимости (interference graph).Каждой переменной соответствует вершина. Если области жизни переменных пересекаются, то соответствующие им вершины соединяются дугой. Нужно раскрасить вершины графа различными цветами, их количество равно количеству свободных регистров так, чтобы вершины, соединенные дугами, имели разные цвета.

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

Интуитивно понятно, что наибольший успех при распределении регистров будет достигнут, если в регистрах хранятся наиболее часто используемые данные. Т.е. информация, собираемая профилировщиками, крайне необходима и в этом случае.

Page 17: Статическое и динамическое профилирование

На предыдущих лекциях поднималась тема зависимостей. Зависимости используются и рассчитываются для того, чтобы доказать правомерность перестановочных оптимизаций.

При кодогенерации возникает другой вариант использования зависимостей – для определения возможностей повторного использования данных при вычислениях. Это позволяет избежать ненужных загрузок из памяти и сохранения в память.

DO I = 1,N A(I+1) = A(I) +F(…) END DO Имеет смысл t <-> A(I+1), чтобы при следующей

итерации не загружать A(I) из памяти.

Page 18: Статическое и динамическое профилирование

Планирование инструкций (instruction scheduling)

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

Процессор имеет собственный механизм для планирования инструкций и распределения их по исполняющим устройствам. Этот механизм предусматривает упреждающий просмотр поступающих инструкций. Но он может быть недостаточно эффективен, поскольку «окно упреждающего просмотра» ограничено. Инструкции могут переставляться в соответствии со следующими соображениями:

• Вынос чтения из памяти как можно дальше от использования результатов чтения. • Смешение инструкций, использующих разные исполняемые устройства

процессора.• Сближение инструкций, использующих одну и ту же переменную (для упрощения

выделения регистров). Планирование инструкций может производиться внутри одного базового блока или

внутри суперблока, объединения нескольких базовых блоков. Некоторые инструкции могут передвигаться за границы соответствующих им базовых блоков.

Планирование инструкций может осуществляться как до, так и после распределения регистров.

Page 19: Статическое и динамическое профилирование

Пример процессорно-архитектурной оптимизации (использование cmovne)

С помощью условных присвоений зависимость по исполнению (control flow dependence) заменяется на зависимость по данным. Ветвление исчезает, и это ускоряет работу с плохо прогнозируемыми переходами.

#include <stdio.h>int main() {int volatile t1,t2,t3;int i,j,aa;int a[1000];t1=t2=t3=0;aa=0; for(i=1;i<100000;i++) { for(j=1;j<1000;j++){ if(t1|t2|t3) aa=2; else aa=0; a[j]=a[j]+aa; t3=j%2; }} printf("%d\n",a[50]);}

Page 20: Статическое и динамическое профилирование

Скомпилируем программу с помощью ia32 компилятора:

icc test.c -O2 -xP -o a.out time ./a.out 0m0.379s

icc test.c -O2 -o b.out time ./b.out 0m0.441s

Ассемблерный код для первого случая:

..B1.3: # Preds ..B1.9 ..B1.2

movl 4008(%esp), %ebx #12.7

orl 4004(%esp), %ebx #12.10

movl $2, %edx #15.6

orl 4000(%esp), %ebx #12.13

movl $0, %ebx #15.6

cmovne %edx, %ebx #15.6

addl %ebx, (%esp,%eax,4) #16.14

movl %eax, %edx #17.9

andl $-2147483647, %edx #17.9

jge ..B1.9 # Prob 50% #17.9

# LOE eax edx ecx esi edi

..B1.10: # Preds ..B1.3

subl $1, %edx #17.9

orl $-2, %edx #17.9

addl $1, %edx #17.9

# LOE eax edx ecx esi edi

..B1.9: # Preds ..B1.3 ..B1.10

movl %edx, 4000(%esp) #17.4

addl $1, %eax #11.17

cmpl $1000, %eax #11.12

jl ..B1.3

Page 21: Статическое и динамическое профилирование

Код для второго случая (без условного присвоения)..B1.3: # Preds ..B1.9 ..B1.2 movl 4008(%esp), %ecx #12.7 orl 4004(%esp), %ecx #12.10 orl 4000(%esp), %ecx #12.13 movl $2, %ecx #15.6 jne ..L1 # Prob 50% #15.6 movl $0, %ecx #15.6..L1: # addl %ecx, (%esp,%edx,4) #16.14 movl %edx, %ecx #17.9 andl $-2147483647, %ecx #17.9 jge ..B1.9 # Prob 50% #17.9 # LOE eax edx ecx ebx esi edi..B1.10: # Preds ..B1.3 subl $1, %ecx #17.9 orl $-2, %ecx #17.9 addl $1, %ecx #17.9 # LOE eax edx ecx ebx esi edi..B1.9: # Preds ..B1.3 ..B1.10 movl %ecx, 4000(%esp) #17.4 addl $1, %edx #11.17 cmpl $1000, %edx #11.12 jl ..B1.3

Page 22: Статическое и динамическое профилирование

10/17/10

Спасибо за внимание!