Анализ петлевой зависимости - Loop dependence analysis - Wikipedia

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

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

Описание

Анализ петлевой зависимости происходит на нормализованный цикл формы:


для меня1 до U1 сделать для меня2 до U2 делать ... для меняп до Uп делать тело    сделано сделано


куда тело может содержать:


S1 a [f11, ..., яп), ..., fм1, ..., яп)]: = ... ... S2 ...: = a [h11, ..., яп), ..., hм1, ..., яп)]


Где а является m-мерным массивом и жп, часпи т. д. - отображение функций из всех индексов итераций (iп) к доступу к памяти в определенном измерении массива.

Например, в C:

за (я = 0; я < U1; я++)  за (j = 0; j < U2; j++)    а[я+4-j] = б[2*я-j] + я*j;

ж1 было бы я + 4-j, управляя записью в первом измерении а и ч2 было бы 2 * i-j, контролируя чтение по первому измерению б.

Задача состоит в том, чтобы найти все возможные зависимости между S1 и S2. Чтобы быть консервативным, любая зависимость, ложность которой не может быть доказана, должна считаться истинной.

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

В ходе доказательства (опровержения) таких зависимостей утверждение S могут быть разложены в зависимости от того, на какой итерации он исходит. Например, S[1,3,5] относится к итерации, где i1 = 1, i2 = 3 и i3 = 5. Конечно, ссылки на абстрактные итерации, такие как S[d1+1,d2,d3], разрешены и распространены.

Зависимость от данных

Зависимости данных показывают отношения между переменными в коде. Есть три разных типа зависимостей данных:

  • Истинная зависимость (иногда называемая зависимостью от потока)
  • Анти-зависимость
  • Выходная зависимость

Истинная зависимость

А истинная зависимость происходит, когда место в памяти записывается до чтения.[1][2][3] Он вводит опасность чтения после записи (RAW) потому что инструкция, которая читает из места в памяти, должна ждать, пока она не будет записана предыдущей инструкцией, иначе инструкция чтения прочитает неправильное значение.[2] Пример истинной зависимости:

 S1: а = 5; S2: б = а;

В этом примере существует истинная зависимость между S1 и S2, потому что переменная a сначала записывается в операторе S1, а затем переменная a считывается оператором S2. Эта истинная зависимость может быть представлена ​​как S1 → T S2.

Истинную зависимость также можно увидеть при чтении и записи между разными итерациями в цикле. В следующем примере показана истинная зависимость между разными итерациями.

 за(j = 1; j < п; j++)    S1: а[j] = а[j-1];

В этом примере истинная зависимость существует между оператором S1 на j-й итерации и S1 на j + 1-й итерации. Существует истинная зависимость, потому что значение будет записано в [j] в одной итерации, а затем будет выполнено чтение [j-1] в следующей итерации. Эта истинная зависимость может быть представлена ​​как S1 [j] → T S1 [j + 1].

Анти-зависимость

An анти-зависимость происходит, когда место в памяти считывается до того, как в это место записывается.[1][2][3] Это вводит опасность записи после чтения (WAR) потому что инструкция, которая записывает данные в ячейку памяти, должна ждать, пока эта ячейка памяти не будет прочитана предыдущей инструкцией, иначе инструкция чтения прочитала бы неправильное значение.[2] Пример антизависимости:

 S1: а = б; S2: б = 5;

В этом примере существует противоположная зависимость между операторами S1 и S2. Это антизависимость, потому что переменная b сначала считывается в операторе S1, а затем переменная b записывается в операторе S2. Это может быть представлено как S1 → A S2. Анти-зависимость можно увидеть в разных итерациях цикла. В следующем примере показан пример этого случая:

 за(j = 0; j < п; j++)    S1: б[j] = б[j+1];

В этом примере существует анти-зависимость между j-й итерацией S1 и j + 1-м элементом S1. Здесь j + 1-й элемент считывается до того, как тот же элемент будет записан в следующей итерации j. Эту антизависимость можно представить как S1 [j] → A S1 [j + 1].

Выходная зависимость

An выходная зависимость происходит, когда место в памяти записывается до того, как это же место снова записывается в другом операторе.[1][2][3] Это вводит опасность записи после записи (WAW) потому что вторая инструкция для записи значения в ячейку памяти должна ждать, пока первая инструкция закончит запись данных в ту же ячейку памяти, иначе, когда ячейка памяти будет считана позже, она будет содержать неправильное значение.[2] Пример выходной зависимости:

  S1: c = 8;   S2: c = 15;

В этом примере есть выходная зависимость между операторами S1 и S2. Здесь переменная c сначала записывается в S1, а затем переменная c снова записывается в операторе S2. Эту выходную зависимость можно представить как S1 → O S2. Выходную зависимость можно увидеть на разных итерациях цикла. В следующем фрагменте кода показан пример этого случая:

 за(j = 0; j < п; j++)    S1: c[j] = j;      S2: c[j+1] = 5;

В этом примере существует выходная зависимость между j-м элементом в S1 и j + 1-м элементом в S2. Здесь c [j + 1] в операторе S2 записывается за одну итерацию. На следующей итерации c [j] в операторе S2, который является той же ячейкой памяти, что и c [j + 1] на предыдущей итерации, записывается снова. Эту выходную зависимость можно представить как S1 [j] → O S2 [j + 1].

Зависимость управления

Зависимости управления также необходимо учитывать при анализе зависимостей между различными операторами в цикле. Зависимости управления - это зависимости, вводимые кодом или самим алгоритмом программирования. Они контролируют порядок, в котором инструкции выполняются при выполнении кода.[4] Один из распространенных примеров - это оператор «если». Операторы if создают в программе ответвления. Часть «then» в выражении «if» явно направляет или контролирует действия, которые необходимо предпринять.[3]

 // Блок кода 1 (ПРАВИЛЬНО) // Блок кода 2 (НЕПРАВИЛЬНО) // Блок кода 3 (НЕПРАВИЛЬНО) если (а == б) тогда {                 если (а == б) тогда {                 если (а == б) тогда {                   c = "контролируемый";               }                                     c = "контролируемый"; }                                  c = "контролируемый";                     d = "не контролируется"; d = "не контролируется";              d = "не контролируется";              }

В этом примере проиллюстрированы ограничения на поток управления. Блок кода 1 показывает правильный порядок при использовании оператора if на языке программирования C. Блок кода 2 иллюстрирует проблему, когда оператор, который должен управляться оператором if, больше не управляется им. Блок кода 3 иллюстрирует проблему, когда оператор, который не должен управляться оператором «if», теперь перемещен под его управление. Обе эти возможности могут привести к неправильному выполнению программы, и их необходимо учитывать при распараллеливании этих операторов в цикле.

Петлевые зависимости против петлезависимой зависимости

Зависимости, переносимые с помощью цикла, и зависимости, не зависящие от цикла, определяются отношениями между операторами в итерациях цикла. Когда оператор в одной итерации цикла каким-то образом зависит от оператора в другой итерации того же цикла, существует переносимая циклом зависимость.[1][2][3] Однако, если оператор в одной итерации цикла зависит только от оператора в той же итерации цикла, это создает независимую от цикла зависимость.[1][2][3]

 // Кодовый блок 1 // Кодовый блок 2 за (я = 0; я < 4; я++)                               за (я = 0; я < 4; я++)    S1: б[я] = 8;                                           S1: б[я] = 8;    S2: а[я] = б[я-1] + 10;                                 S2: а[я] = б[я] + 10;

В этом примере кодовый блок 1 показывает зависящую от цикла зависимость между итерацией i-го оператора S2 и итерацией i-1 оператора S1. Это означает, что оператор S2 не может выполняться, пока не завершится оператор S1 в предыдущей итерации. Блок кода 2 показывает независимую от цикла зависимость между операторами S1 и S2 в одной итерации.

Графики циклических зависимостей и обхода итерационного пространства

Графы обхода пространства итераций (ITG) показывают путь, который проходит код при обходе итераций цикла.[1] Каждая итерация представлена ​​узлом. Графики зависимостей с переносом по циклу (LDG) дают визуальное представление всех истинных зависимостей, антизависимостей и зависимостей вывода, которые существуют между различными итерациями в цикле.[1] Каждая итерация представлена ​​узлом.

Разницу между двумя графиками легче показать с помощью вложенного цикла for.

 за (я = 0; я < 4; я++)    за (j = 0; j < 4; j++)       S1: а[я][j] = а[я][j-1] * Икс;

В этом примере существует истинная зависимость между j-й итерацией оператора S1 и j + 1-м оператором S1. Это можно представить как S1 [i, j] → T S1 [i, j + 1] График обхода итерационного пространства и граф зависимостей с переносом цикла: График обхода пространства итераций: График зависимостей с переносом цикла:

График зависимостей с переносом петель (LDG)
Граф обхода пространства итераций (ITG)


Смотрите также

Рекомендации

  1. ^ а б c d е ж грамм Солихин, Ян (2016). Основы параллельной компьютерной архитектуры: многокристальные и многоядерные системы. [Соединенные Штаты?]: Паб Солихин. ISBN  978-1-4822-1118-4.
  2. ^ а б c d е ж грамм час Деван, Прадип; Камат, Р.К. (2014). «Обзор - Анализ зависимостей LOOP для распараллеливающего компилятора». Международный журнал компьютерных наук и информационных технологий. 5.
  3. ^ а б c d е ж Джон, Хеннесси; Паттерсон, Дэвид (2012). Компьютерная архитектура - количественный подход. 225 Wyman Street, Waltham, MA 02451, США: Morgan Kaufmann Publishers. С. 152–156. Дои:10.1016 / B978-0-12-383872-8.00003-3 (неактивно 11.11.2020). ISBN  978-0-12-383872-8.CS1 maint: location (связь) CS1 maint: DOI неактивен по состоянию на ноябрь 2020 г. (связь)
  4. ^ Allen, J. R .; Кеннеди, Кен; Портерфилд, Кэрри; Уоррен, Джо (1983-01-01). «Преобразование зависимости управления в зависимость от данных». Материалы 10-го симпозиума ACM SIGACT-SIGPLAN по принципам языков программирования. ПОПЛ '83. Нью-Йорк, Нью-Йорк, США: ACM: 177–189. Дои:10.1145/567067.567085. ISBN  0897910907. S2CID  39279813.