C $Header: /u/gcmpack/MITgcm/eesupp/src/scatter_2d_rx.template,v 1.7 2012/09/03 19:36:29 jmc Exp $ C $Name: $ #include "PACKAGES_CONFIG.h" #include "CPP_EEOPTIONS.h" CBOP C !ROUTINE: SCATTER_2D_RX C !INTERFACE: SUBROUTINE SCATTER_2D_RX( I gloBuff, O myField, I xSize, ySize, I useExch2GlobLayOut, I zeroBuff, I myThid ) C !DESCRIPTION: C Scatter elements of a global 2-D array from mpi process 0 to all processes. C Note: done by Master-Thread ; might need barrier calls before and after C this S/R call. C !USES: IMPLICIT NONE #include "SIZE.h" #include "EEPARAMS.h" #include "EESUPPORT.h" #ifdef ALLOW_EXCH2 #include "W2_EXCH2_SIZE.h" #include "W2_EXCH2_TOPOLOGY.h" #endif /* ALLOW_EXCH2 */ C !INPUT/OUTPUT PARAMETERS: C gloBuff ( _RX ) :: full-domain 2D IO-buffer array (Input) C myField ( _RX ) :: tiled, local (i.e. my Proc. tiles) 2D array (Output) C xSize (integer):: global buffer 1rst dim (x) C ySize (integer):: global buffer 2nd dim (y) C useExch2GlobLayOut:: =T: Use Exch2 global-map layout (only with EXCH2) C zeroBuff (logical):: =T: reset the buffer to zero after copy C myThid (integer):: my Thread Id number INTEGER xSize, ySize _RX gloBuff(xSize,ySize) _RX myField(1:sNx,1:sNy,nSx,nSy) LOGICAL useExch2GlobLayOut LOGICAL zeroBuff INTEGER myThid CEOP C !LOCAL VARIABLES: INTEGER i,j, bi,bj INTEGER iG, jG INTEGER iBase, jBase #ifdef ALLOW_EXCH2 INTEGER iGjLoc, jGjLoc INTEGER tN #endif /* ALLOW_EXCH2 */ #ifdef ALLOW_USE_MPI INTEGER np, pId _RX temp(1:sNx,1:sNy,nSx,nSy) INTEGER istatus(MPI_STATUS_SIZE), ierr INTEGER lbuff, isource, itag #endif /* ALLOW_USE_MPI */ _BEGIN_MASTER( myThid ) #ifdef ALLOW_USE_MPI IF ( usingMPI ) THEN lbuff = sNx*nSx*sNy*nSy isource = 0 itag = 0 IF( mpiMyId .EQ. 0 ) THEN C-- Process 0 sends local arrays to all other processes DO np = 2, nPx*nPy C-- Process 0 extract the local arrays from the global buffer. #ifdef ALLOW_EXCH2 IF ( useExch2GlobLayOut ) THEN DO bj=1,nSy DO bi=1,nSx tN = W2_procTileList(bi,bj,np) IF ( exch2_mydNx(tN) .GT. xSize ) THEN C- face x-size larger than glob-size : fold it iGjLoc = 0 jGjLoc = exch2_mydNx(tN) / xSize ELSEIF ( exch2_tNy(tN) .GT. ySize ) THEN C- tile y-size larger than glob-size : make a long line iGjLoc = exch2_mydNx(tN) jGjLoc = 0 ELSE C- default (face fit into global-IO-array) iGjLoc = 0 jGjLoc = 1 ENDIF DO j=1,sNy #ifdef TARGET_NEC_SX !cdir novector #endif iG=exch2_txGlobalo(tN)+iGjLoc*(j-1)-1 jG=exch2_tyGlobalo(tN)+jGjLoc*(j-1) DO i=1,sNx temp(i,j,bi,bj) = gloBuff(iG+i,jG) ENDDO ENDDO ENDDO ENDDO ELSE #else /* ALLOW_EXCH2 */ IF (.TRUE.) THEN #endif /* ALLOW_EXCH2 */ iBase = mpi_myXGlobalLo(np)-1 jBase = mpi_myYGlobalLo(np)-1 DO bj=1,nSy DO bi=1,nSx DO j=1,sNy #ifdef TARGET_NEC_SX !cdir novector #endif iG = iBase+(bi-1)*sNx jG = jBase+(bj-1)*sNy+j DO i=1,sNx temp(i,j,bi,bj) = gloBuff(iG+i,jG) ENDDO ENDDO ENDDO ENDDO C end if-else useExch2GlobLayOut ENDIF C-- Process 0 sends local arrays to all other processes pId = np - 1 CALL MPI_SEND (temp, lbuff, _MPI_TYPE_RX, & pId, itag, MPI_COMM_MODEL, ierr) C- end loop on np ENDDO ELSE C-- All proceses except 0 receive local array from process 0 CALL MPI_RECV (myField, lbuff, _MPI_TYPE_RX, & isource, itag, MPI_COMM_MODEL, istatus, ierr) ENDIF ENDIF #endif /* ALLOW_USE_MPI */ IF( myProcId .EQ. 0 ) THEN C-- Process 0 fills-in its local data #ifdef ALLOW_EXCH2 IF ( useExch2GlobLayOut ) THEN DO bj=1,nSy DO bi=1,nSx tN = W2_myTileList(bi,bj) IF ( exch2_mydNx(tN) .GT. xSize ) THEN C- face x-size larger than glob-size : fold it iGjLoc = 0 jGjLoc = exch2_mydNx(tN) / xSize ELSEIF ( exch2_tNy(tN) .GT. ySize ) THEN C- tile y-size larger than glob-size : make a long line iGjLoc = exch2_mydNx(tN) jGjLoc = 0 ELSE C- default (face fit into global-IO-array) iGjLoc = 0 jGjLoc = 1 ENDIF DO j=1,sNy #ifdef TARGET_NEC_SX !cdir novector #endif iG=exch2_txGlobalo(tN)+iGjLoc*(j-1)-1 jG=exch2_tyGlobalo(tN)+jGjLoc*(j-1) DO i=1,sNx myField(i,j,bi,bj) = gloBuff(iG+i,jG) ENDDO ENDDO ENDDO ENDDO C-- After the copy from the buffer, reset to zero. C An alternative to zeroBuff when writing to file, C which could be faster if we do less read than write. IF ( zeroBuff ) THEN DO j=1,ySize DO i=1,xSize gloBuff(i,j) = 0. ENDDO ENDDO ENDIF ELSE #else /* ALLOW_EXCH2 */ IF (.TRUE.) THEN #endif /* ALLOW_EXCH2 */ iBase = myXGlobalLo-1 jBase = myYGlobalLo-1 DO bj=1,nSy DO bi=1,nSx DO j=1,sNy #ifdef TARGET_NEC_SX !cdir novector #endif iG = iBase+(bi-1)*sNx jG = jBase+(bj-1)*sNy+j DO i=1,sNx myField(i,j,bi,bj) = gloBuff(iG+i,jG) ENDDO ENDDO ENDDO ENDDO C end if-else useExch2GlobLayOut ENDIF C- end if myProcId = 0 ENDIF _END_MASTER( myThid ) RETURN END C---+----1----+----2----+----3----+----4----+----5----+----6----+----7-|--+----|