Сложные типы с использованием OpenMP - PullRequest
0 голосов
/ 16 февраля 2012

Я новичок в использовании OpenMP с C ++.Я делал простую функцию, загружая два массива, используя цикл for.Эти массивы определены как сложные.

#include <omp.h>
#include <iostream>
#include <stdlib.h>
#include <complex>
#define CHUNKSIZE   10
#define N       100

using namespace std;

int main (int argc, char *argv[]) 
{
int nthreads, tid, i, chunk;
complex<double> a[N], b[N], c[N];

/* Some initializations */
for (i=0; i < N; i++)
    a[i].real() = b[i].real() = i * 1.0;
chunk = CHUNKSIZE;

#pragma omp parallel shared(a,b,c,nthreads,chunk) private(i,tid)
{
    tid = omp_get_thread_num();
    if (tid == 0)
    {
        nthreads = omp_get_num_threads();
        printf("Number of threads = %d\n", nthreads);
    }
    printf("Thread %d starting...\n",tid);

#pragma omp for schedule(dynamic,chunk)
    for (i=0; i<N; i++)
    {
        c[i] = a[i] + b[i];
        printf("Thread %d: c[%d]= %e\n",tid,i,c[i]);
    }

}  /* end of parallel section */

}

При компиляции я получаю следующее предупреждение:

omp_complex.cpp: 43: warning: не может передавать объекты не-POD типа struct std :: complex 'through'.. ';вызов будет прерван во время выполнения

, и если я запусту a.out, я получу сообщение «Недопустимая инструкция» на экране.Я пытался выяснить, что происходит, но я не нашел хорошей ссылки.Кто-нибудь знает, разрешены ли сложные типы в директивах OpenMP?

Ответы [ 2 ]

4 голосов
/ 16 февраля 2012

Ошибка здесь:

printf("Thread %d: c[%d]= %e\n",tid,i,c[i]);

printf не знает (и не может знать), как обращаться с std::complex.Используйте потоковые операции C ++ для вывода сложных типов.

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

Я обычно использую макрос для этого (но C ++ 11 облегчает это с помощью шаблонов с переменными значениями):

#define THREAD_SAFE_OUT(out, message) \
    do { \
        std::ostringstream buffer; \
        buffer << message; \
        out << buffer.str(); \
    while (false)

…

THREAD_SAFE_OUT(std::cout, "Thread " << tid << ": c[" << i << "] = " << c[i]);

Слово о стиле при использовании OpenMP:

Не используйте директиву private.Это просто обходной путь для языков, которые требуют, чтобы все переменные были объявлены в начале метода.Поскольку C ++ не требует этого, лучше (всегда) объявлять переменные при первом использовании, то есть внутри параллельного раздела.Таким образом, они также являются приватными.

2 голосов
/ 16 февраля 2012

Ваша проблема заключается в передаче std::complex<double> в printf, что, как говорит вам компилятор, не будет работать.Как вы хотите, чтобы оно отображалось в любом случае?

Вы должны печатать реальные и мнимые части отдельно, или норму и фазу, если хотите.

...