Поделиться через


Ошибка компилятора C3020

Var: переменная индекса цикла OpenMP for не может быть изменена в тексте цикла

Цикл OpenMP for не может изменять индекс (счетчик циклов) в тексте for цикла.

Следующий пример приводит к возникновению ошибки C3020:

// C3020.cpp
// compile with: /openmp
int main() {
   int i = 0, n = 3;

   #pragma omp parallel
   {
      #pragma omp for
      for (i = 0; i < 10; i += n)
         i *= 2;   // C3020
         // try the following line instead
         // n++;
   }
}

Переменная, объявленная с lastprivate , не может использоваться в качестве индекса внутри параллелизованного цикла.

Следующий пример даст C3020 для второго последнегоprivate, так как последнийprivate активирует запись в idx_a в самом внешнем цикле. Первый последнийprivate не дает ошибку, так как эта последняя итерация активирует запись в idx_a за пределами внешнего цикла (технически, в самом конце последней итерации). В следующем примере возникает ошибка C3020.

// C3020b.cpp
// compile with: /openmp /c
float a[100][100];
int idx_a, idx_b;
void test(int first, int last)
{
   #pragma omp parallel for lastprivate(idx_a)
   for (idx_a = first; idx_a <= last; ++idx_a) {
      #pragma omp parallel for lastprivate(idx_a)   // C3020
      for (idx_b = first; idx_b <= last; ++idx_b) {
         a[idx_a][idx_b] += 1.0f;
      }
   }
}

В следующем примере показано возможное решение:

// C3020c.cpp
// compile with: /openmp /c
float a[100][100];
int idx_a, idx_b;
void test(int first, int last)
{
   #pragma omp parallel for lastprivate(idx_a)
   for (idx_a = first; idx_a <= last; ++idx_a) {
      #pragma omp parallel for lastprivate(idx_b)
      for (idx_b = first; idx_b <= last; ++idx_b) {
         a[idx_a][idx_b] += 1.0f;
      }
   }
}