Матрично-векторное умножение OpenMP на лету в Фортране

Я пишу код на Фортране для выполнения умножения матрицы на вектор. Умножение выполняется на лету в том смысле, что матричные элементы не сохраняются заранее. Код работает хорошо в последовательном случае, но когда я реализую параллельный случай с OpenMP, как показано ниже, я не могу получить правильный вывод. Может кто-нибудь подсказать, что я сделал не так?

Module module1
   integer:: N, M
   integer, allocatable:: Jeven_A(:), Jeven_B(:), Jodd_A(:), Jodd_B(:), s(:)
END module module1

subroutine lookup(a,Pswitch,sa)
    use module1
    implicit none
    integer, intent(in):: a, Pswitch
    integer, intent(out):: sa
    integer:: Ia, Ib

    Ib = ibits(a,N/2,N/2)
    Ia = ibits(a,0,N/2)
    if (Pswitch==0) then
        sa = Jeven_A(Ia)+Jeven_B(Ib)
    else if (Pswitch==1) then
        sa = Jodd_A(Ia)+Jodd_B(Ib)
    end if
end subroutine lookup

SUBROUTINE Mv_onthefly(V, vni, vno)
    USE module1
    IMPLICIT NONE
    INTEGER, INTENT(IN):: vni, vno
    complex, DIMENSION(1:M,1:3):: V
    INTEGER::sb, i,j ,k, a, b
    complex, allocatable:: Vvni(:), Vvno(:)

    allocate(Vvni(1:M),Vvno(1:M))
    Do i=1, M
        Vvni(i) = V(i,vni)
    END DO
    Vvno = 0.

!$OMP PARALLEL private(k,i,j,sb,a,b,bittest,Vvni)
!$OMP DO REDUCTION(+:Vvno)
   DO k=1,M
       a = s(k) 
       b = 0
       DO i=0, N-1
            Do j=0, N-1
                  b = IEOR(a,2**i+2**j)
                  CALL lookup(b,0,sb)  !subroutine which lookup the index sb corresponds to b in the array s
                  Vvno(sb)=Vvno(sb)-J1*Vvni(k)
            END do
        END DO
        DO i=0, N-1
            Vvno(k)=Vvno(k)-J2*Vvni(k)
        END DO
    END DO
!$OMP END DO
!$OMP END PARALLEL

   Do i =1, M
       V(i,vno)=Vvno(i)
   END DO
   deallocate(Vvni,Vvno)

END SUBROUTINE Mv_onthefly

0 ответов

Другие вопросы по тегам