Вычислить одно значение в процессе и отправить его в корневой процесс mpi - PullRequest
0 голосов
/ 28 февраля 2019

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

вот мой псевдокод:

*calculates which process will compute the value.

if (rank == proc) then

*do some operations and calculate the value
! if the process is not the root, send the value.
  if (proc /= 0) 
    call MPI_SEND(auxvalue,1,MPI_REAL8,0,1,MPI_COMM_WORLD,ierr)
  end if
else
  if (size>1) then
    if (rank==0) then
     call MPI_RECV(auxvalue,1,MPI_REAL8,proc,1,MPI_COMM_WORLD,status,ierr)
    end if
  end if

end if

, но корневой процесс ни не получает значение, ни процесс никогда не отправляетit.

Можете ли вы помочь мне?

edit:

, как вы предлагаете, здесь код, это только подпрограмма, которая интерполирует (используя бикубический метод) значениефункции f в точке.Начальные функции mpi находятся в основном рутине, эта часть только интерполирует функцию.Модули, которые я здесь использую, работают хорошо и проверены.

real (8) функция bicuinterp (r0, z0, f, Drf, Dzf, Drzf, errorflag)

use mpi
use procinfo
use param
use arrays

implicit none

integer status(MPI_STATUS_SIZE)
logical errorflag       ! Flag to control the behaviour of error.
real(8),dimension(1-ghost:Nrmax,1-ghost:Nzmax) :: f,Drf,Dzf,Drzf
                                                ! Function and his derivatives to be interpolated.
real(8),dimension (1:4,1:4) :: M, fval, auxA
real(8),dimension (4) :: x,y
                                                ! Powers of grid point distances.
real(8) r0,z0                   ! Point at which we are interpolating.
real(8) deltar,deltaz       ! Distance to the grid point to the left.
real(8) Nauxr, Nauxz        ! Coordinates of the processor which owns the grid point.
real(8) rmin,rmax           ! Minimum and maximum rho-coordinate.
real(8) zmin,zmax           ! Minimum and maximum z-coordinate.
real(8) drdz                ! Product of dr*dz
real(8) small
real(8) auxbicuinterp
integer procr, procz, proc ! Processor which owns the grid point.
integer i0,j0              ! Grid point immediatly to the left in rho,z directions.
integer i,j,k,l            ! Counters

! Common constants
drdz = dr*dz

! ************************
! ***   SANITY CHECK   ***
! ************************

! First check that the interpolating point is inside the grid,
! and that there are enough points to either side for bicubic
! interpolation.

    rmin = (dble(1-ghost)-0.5d0)*dr
    rmax = (dble(Nrtotal)-0.5d0)*dr
if (eqsym) then
    zmin = (dble(1-ghost)-0.5d0)*dz
    zmax = (dble(Nztotal)-0.5d0)*dz
else
    zmin = (-dble(Nztotal)/2.0+0.5d0)*dz
    zmax = (dble(Nztotal)/2.0-0.5d0)*dz
end if

if (rank==0) then

    if ((r0<rmin).or.(r0>rmax) .or. (z0<zmin) .or. (z0>zmax)) then
        print *
        print *, 'interp.f90: Interpolating point is outside the grid'
        write(*,'(A,ES12.4)') ' Interpolating position =',r0,',',z0
        write(*,'(A,ES12.4)') ' Left boundary          =',rmin
        write(*,'(A,ES12.4)') ' Right boundary         =',rmax
        write(*,'(A,ES12.4)') ' Lower boundary          =',zmin
        write(*,'(A,ES12.4)') ' Upper boundary         =',zmax
        write(*,'(A,I3)') ' Processor  =',rank
        print *
        print *, 'Aborting ...'
        print *
        bicuinterp = 0.0D0
        go to 200
        !           call die
    else if (r0<rmin+dr) then
        print *
        print *, 'interp.f90: Not enough points to left side for cubic interpolation'
        write(*,'(A,ES12.4)') ' Interpolating position =',r0
        write(*,'(A,ES12.4)') ' Left boundary          =',rmin
        write(*,'(A,ES12.4)') ' Grid spacing           =',dr
        print *
        print *, 'Aborting ...'
        print *
        bicuinterp = 0.0D0
        !     call die
        go to 200
    else if (r0>rmax-dr) then
        print *
        print *, 'interp.f90: Not enough points to right side for cubic interpolation'
        write(*,'(A,ES12.4)') ' Interpolating position =',r0
        write(*,'(A,ES12.4)') ' Right boundary         =',rmax
        write(*,'(A,ES12.4)') ' Grid spacing           =',dr
        print *
        print *, 'Aborting ...'
        print *
        bicuinterp = 0.0D0
        go to 200
        !           call die
    else if (z0<zmin+dz) then
        print *
        print *, 'interp.f90: Not enough points to lower side for cubic interpolation'
        write(*,'(A,ES12.4)') ' Interpolating position =',z0
        write(*,'(A,ES12.4)') ' Left boundary          =',zmin
        write(*,'(A,ES12.4)') ' Grid spacing           =',dz
        print *
        print *, 'Aborting ...'
        print *
        bicuinterp =0.0D0
        go to 200
        !           call die
    else if (z0>zmax-dz) then
        print *
        print *, 'interp.f90: Not enough points to lower side for cubic interpolation'
        write(*,'(A,ES12.4)') ' Interpolating position =',z0
        write(*,'(A,ES12.4)') ' Left boundary          =',zmax
        write(*,'(A,ES12.4)') ' Grid spacing           =',dz
        print *
        print *, 'Aborting ...'
        print *
        bicuinterp = 0.0D0
        go to 200
        !           call die
    end if
end if

! **************************************
! ***   CHECK IF YOU OWN THE POINT   ***
! **************************************

! For multiple processor runs check that you
! own the point to be interpolated and that
! there are enough points for bicubic interpolation.
!
! Notice that we need to make sure that only one
! processor owns the point, which might be a problem
! close to inter-processor boundaries where there is
! an overlap.  When that happens here I choose the
! processor to the left as owner of the point (but
! carefull with processor zero that has no neighbour
! to the left).

if (size>1) then

    !    If you don't own the grid point then either die,
    !    or return 0.

    if ((r0<=r(Nminr(rank),1)).or.(r0>r(Nmaxr(rank),1)) .or.(z0<=z(1,Nminz(rank))).or.(z0>z(1,Nmaxz(rank)))) then

        if (errorflag) then
            write(*,'(A,3ES12.4)') ' bicuinterp.f90 I do not own the point (r0,z0,rmin,rmax,zmin,zmax):',r0,rmin,rmax
            write(*,'(A,I3,A,I3)') ' Processor:',rank
            print *, 'Aborting ...'
            !           call die
            print *
        else
            auxbicuinterp = 0.d0
            !        return
        end if

    end if

end if

! *********************************************
! ***   FIND GRID POINT TO THE LEFT OF r0   ***
! *********************************************

! Find the processor who owns the grid point.

Nauxr = Nrtotal/nprocr
Nauxz = Nztotal/nprocz

procr = int(r0/(dr*Nauxr))!-1
if (eqsym) then
procz = int(z0/(dz*Nauxz))!-1
else
procz = int((z0/dz+0.5*Nztotal)/Nauxz)
end if

proc = procr + nprocr*procz
! If the grid point at which we are interpolating is at the right boundary of
! the processor, we take the next processor to the right.

if (Nmaxr(proc)*dr<=r0+dr) then
    procr = procr+1
end if
if (Nmaxz(proc)*dz<=z0+dr) then
    procz = procz+1
end if

proc = procr + nprocr*procz

! Find the grid point to the left to our point (rho,z)
if (rank==proc) then

i0 = int(r0/dr + 0.5D0-Nminr(proc))

if (eqsym) then
j0 = int(z0/dz + 0.5D0-Nminz(proc))
else
j0 = int(z0/dz -Nminz(proc)+0.5*(Nztotal+1-ghost))
end if
    ! Calculate separation from grid point

    deltar = (r0-r(i0,j0))/dr
    deltaz = (z0-z(i0,j0))/dz

    ! Check that the point to be interpolated
    ! is indeed between grid points (i0,i0+1) and (j0,j0+1).

    small = 1.d-10

    ! if ((deltar<-small).or.(deltar>dr+small)) then
    !
    !   print *
    !   print *, 'bicuinterp.f90: this should never happen! error in deltar'
    !   !        print *, 'deltar', deltar, 'dr+small',dr+small, 'rank',rank
    !   print *, r0,r(i0,j0),r(i0+1,j0),rank,deltar,dr,i0,Nmaxr(proc)*dr
    !   print *, 'Aborting ...'
    !   print *
    !   !       call die
    !   bicuinterp = 0.0D0
    !   goto 200
    ! end if
    ! if ((deltaz<-small).or.(deltaz>dz+small)) then
    !
    !   print *
    !   print *, 'bicuinterp.f90: this should never happen! error in deltaz'
    !   !        print *, 'deltaz', deltaz, 'dz+small',dz+small
    !   print *, z0,z(i0,j0),z(i0,j0+1),rank,deltaz,dz,j0,z0/dz
    !   print *, 'Aborting ...'
    !   print *, 'Nmaxr', Nmaxr(proc), Nminr(proc+1)
    !   print *
    !   !       call die
    !   bicuinterp = 0.0D0
    !   goto 200
    ! end if

    if ( (r(i0,j0)>r0+small) .or. (r0-small>r(i0+1,j0)) ) then
        print *, 'bicuinterp.f90, point outside grid points: ERROR'
        !               print *, 'r0=',r0,' is not between (',i0,i0+1,')'
        !               print *, r(i0,j0),r(i0+1,j0)
        bicuinterp = 0.0D0
        goto 200
    end if
    if ( (z(i0,j0)>z0+small) .or. (z0-small>z(i0,j0+1)) ) then
        print *, 'bicuinterp.f90, point outside grid points: ERROR'
                        print *, 'z0 is not between (j0,j0+1)'

                        print *, z0,z(i0,j0),z(i0,j0+1)
        bicuinterp = 0.0D0
        goto 200
    end if

! ********************************
! ***   BICUBIC INTERPOLATON   ***
! ********************************
    ! Fill the values of the function and his derivative at the corners for interpolation.

    fval(1,1) =    f(i0  ,j0  )
    fval(1,2) =    f(i0  ,j0+1)
    fval(1,3) =  Dzf(i0  ,j0  )*dz
    fval(1,4) =  Dzf(i0  ,j0+1)*dz
    fval(2,1) =    f(i0+1,j0  )
    fval(2,2) =    f(i0+1,j0+1)
    fval(2,3) =  Dzf(i0+1,j0  )*dz
    fval(2,4) =  Dzf(i0+1,j0+1)*dz
    fval(3,1) =  Drf(i0  ,j0  )*dr
    fval(3,2) =  Drf(i0  ,j0+1)*dr
    fval(3,3) = Drzf(i0  ,j0  )*drdz
    fval(3,4) = Drzf(i0  ,j0+1)*drdz
    fval(4,1) =  Drf(i0+1,j0  )*dr
    fval(4,2) =  Drf(i0+1,j0+1)*dr
    fval(4,3) = Drzf(i0+1,j0  )*drdz
    fval(4,4) = Drzf(i0+1,j0+1)*drdz

    !Fill coefficients for matrix interpolation

    M = 0.0D0

    M(1,1) =  1.0D0
    M(2,3) =  1.0D0
    M(3,1) = -3.0D0
    M(3,2) =  3.0D0
    M(3,3) = -2.0D0
    M(3,4) = -1.0D0
    M(4,1) =  2.0D0
    M(4,2) = -2.0D0
    M(4,3) =  1.0D0
    M(4,4) =  1.0D0

    ! Array with powers of the separation from the grid points
    x = 0.0D0
    x(1) = 1.0D0
    x(2) = deltar
    x(3) = deltar**2
    x(4) = deltar**3

    y = 0.0D0
    y(1) = 1.0D0
    y(2) = deltaz
    y(3) = deltaz**2
    y(4) = deltaz**3


    auxbicuinterp = 0.0D0
    do l=1,4
        do k=1,4
            do j=1,4
                do i=1,4
                    auxbicuinterp = auxbicuinterp + x(i)*M(i,j)*fval(j,k)*M(l,k)*y(l)
                end do
            end do
        end do
    end do
!If the process isn't the root, send it.
if (rank /= 0) then
    call MPI_SEND(auxbicuinterp,1,MPI_REAL8,0,1,MPI_COMM_WORLD,ierr)

end if
else
   if (size>1) then
      !Only the root process receive the value
      if (proc /= 0 .and. rank==0) then
        call MPI_RECV(auxbicuinterp,1,MPI_REAL8,proc,1,MPI_COMM_WORLD,status,ierr)

  end if
end if
end if
 bicuinterp = auxbicuinterp
200 continue
return
end function bicuinterp

Я знаюслишком длинный, но, как я объяснил выше, только один процесс интерполирует значение, и мне нужно отправить его корневому процессу, только если процесс не является корневым (нулевым), но мой код не работает.

...