internal error in mpi.F

Problems running VASP: crashes, internal errors, "wrong" results.


Moderators: Global Moderator, Moderator

Post Reply
Message
Author
martin.gmitra
Newbie
Newbie
Posts: 2
Joined: Wed Mar 30, 2022 9:01 am

internal error in mpi.F

#1 Post by martin.gmitra » Mon Dec 26, 2022 8:48 pm

Dear VASP developers and users,

I am encountering a program vasp_ncl stop with the error message: internal error in: mpi.F at line: 1825
when trying to obtain after successful SCF calculation in the next step DOSCAR and PROCAR files for bands unfolding.
I tried also to reduce jobs per node in order to provide more RAM per job. It does not help.

Thank you in advance for any hints.
Best regards,
Martin Gmitra
Institute of Physics, UPJS in Kosice
You do not have the required permissions to view the files attached to this post.

alexey.tal
Global Moderator
Global Moderator
Posts: 320
Joined: Mon Sep 13, 2021 12:45 pm

Re: internal error in mpi.F

#2 Post by alexey.tal » Fri Dec 30, 2022 7:53 pm

Dear Martin,

It looks like this error occurs due to an integer overflow when pDOS is calculated in SPHPRO_FAST.
Do you see this error if you set LORBIT = None?

I am attaching a patch that switches 4-byte to 8-byte integers in this subroutine.
I see that you are using VASP 6.3.0, so you can apply this patch, recompile the code, and try your calculation again.
The system is too big, so I cannot test it myself, but please let me know if the patch does not solve the problem.

Code: Select all

diff --git a/src/sphpro.F b/src/sphpro.F
index f95ed55f..c83c4dfb 100644
--- a/src/sphpro.F
+++ b/src/sphpro.F
@@ -1461,12 +1461,12 @@ DOESI CSUM = (0.0_q, 0.0_q)
       NIS = NIS+T_INFO%NITYP(NT)
       ENDDO typ
 
-      CALLMPI( M_sum_d( WDES%COMM, ION_SUM, LDIMP*T_INFO%NIONS*WDES%NCDIJ))
-      CALLMPI( M_sum_d( WDES%COMM, ION_SUM_DETAIL, LMDIMP*T_INFO%NIONS*WDES%NCDIJ))
+      CALLMPI( M_sum_d8( WDES%COMM, ION_SUM, 1_qi8*LDIMP*T_INFO%NIONS*WDES%NCDIJ))
+      CALLMPI( M_sum_d8( WDES%COMM, ION_SUM_DETAIL, 1_qi8*LMDIMP*T_INFO%NIONS*WDES%NCDIJ))
       IF (LFINAL) THEN
-        CALLMPI( M_sum_d( WDES%COMM, PAR(1,1,1,1,1), WDES%NB_TOT*WDES%NKPTS*LPAR*T_INFO%NIONP*WDES%NCDIJ))
+        CALLMPI( M_sum_d8( WDES%COMM, PAR(1,1,1,1,1), 1_qi8*WDES%NB_TOT*WDES%NKPTS*LPAR*T_INFO%NIONP*WDES%NCDIJ))
         IF (LORBIT>=12 .AND. LORBIT<=14) THEN
-           CALLMPI( M_sum_d( WDES%COMM, PHAS(1,1,1,1,1), LMDIMP*T_INFO%NIONS*WDES%NKPTS*WDES%NB_TOT*WDES%ISPIN*2))
+           CALLMPI( M_sum_d8( WDES%COMM, PHAS(1,1,1,1,1), 1_qi8*LMDIMP*T_INFO%NIONS*WDES%NKPTS*WDES%NB_TOT*WDES%ISPIN*2))
         ENDIF
       ENDIF
 

martin.gmitra
Newbie
Newbie
Posts: 2
Joined: Wed Mar 30, 2022 9:01 am

Re: internal error in mpi.F

#3 Post by martin.gmitra » Wed Jan 04, 2023 9:19 am

Dear Alexey,

Thank you for your suggestion. Let me provide you with the preliminary findings (vasp 6.3.0):
  • setting LORBIT = None is not recognized

    Code: Select all

    |     Error reading item LORBIT from file INCAR.                              |
    |     Error code was IERR= 5 ... . 
    But ignoring the LORBIT in INCAR the vasp_ncl finished correctly
  • applying your patch requires an M_sum_d8 subroutine. I have added a minimal one (see below) in mpi.F, however, it looks to me that there might many more places in the code that would require "long" integers. Anyhow, I am rerunning the calculations and will keep you posted with the result.

    Code: Select all

    !----------------------------------------------------------------------
    !
    ! M_sum_d8:  performs a sum on n double prec numbers
    !  it uses either sumb_d or sumf_d
    !
    !----------------------------------------------------------------------
    
          SUBROUTINE M_sum_d8(COMM, vec, n)
          USE mpimy
          IMPLICIT NONE
    
          TYPE(communic) COMM
          INTEGER*8 n
          REAL(q) vec(n)
    
    ! return ranks that are not in the group
          IF ( COMM%MPI_COMM == MPI_COMM_NULL ) THEN
             RETURN
          ENDIF
    
    #ifdef use_collective_sum
          CALL M_sumb_d(COMM, vec, n)
    #else
          IF ( n>MPI_BLOCK) THEN
             CALL M_sumf_d(COMM, vec, n)
          ELSE
             CALL M_sumb_d(COMM, vec, n)
          ENDIF
    #endif
          END SUBROUTINE M_sum_d8

alexey.tal
Global Moderator
Global Moderator
Posts: 320
Joined: Mon Sep 13, 2021 12:45 pm

Re: internal error in mpi.F

#4 Post by alexey.tal » Thu Jan 05, 2023 5:28 pm

My bad. The M_sum_d8 subroutine has not been merged into the release version of the code. Your implementation of M_sum_d8 calls M_sumf_d, which takes 4-byte counter, so it will probably hit the overflow in M_sumf_d instead of M_sum_d.

I added the necessary routines. Could you please try your calculation with the updated patch?

Code: Select all

diff -Naur  vasp.6.3.0/src/mpi.F ./src/mpi.F
--- vasp.6.3.0/src/mpi.F	2022-01-20 16:10:06.000000000 +0100
+++ ./src/mpi.F	2023-01-04 13:55:47.113559660 +0100
@@ -4378,6 +4378,137 @@
 
 !----------------------------------------------------------------------
 !
+! M_sumf_d8: performs a fast global sum on n doubles in
+! vector vec (algorithm by Kresse Georg); this a special version
+! for giant arrays exceeding an element count n of 2^31-1 (the
+! maximum size that can be handled with 32-bit INTEGER). Mainly,
+! the major difference to routine M_sumf_d is that the last
+! argument ("n") is now declared as 64-bit INTEGER -- and that
+! in addition some intermediate integer operation need also to
+! be done with 64-bit INTEGER temporary variables (adapted by jF)
+!
+!----------------------------------------------------------------------
+
+      SUBROUTINE M_sumf_d8(COMM, vec, n)
+      USE mpimy
+      USE string, ONLY: str
+      USE tutor, ONLY: vtutor
+      IMPLICIT NONE
+
+      TYPE(communic) COMM
+! this must all declared as 64-bit INTEGER
+      INTEGER(qi8) n,nsummed,ndo,i,j,n_,mmax
+! this must stay to be 32-bit INTEGER and a new "n4" is needed for MPI calls
+      INTEGER ncount,info,n4
+      REAL(q) vec(n)
+!----------------------------------------------------------------------
+#if defined(T3D_SMA)
+!----------------------------------------------------------------------
+      INTEGER MALLOC_DONE
+      INTEGER, EXTERNAL :: ISHM_CHECK
+      COMMON /SHM/ MALLOC_DONE, PBUF
+      POINTER ( PBUF, vec_inter )
+      REAL(q) :: vec_inter(n/COMM%NCPU*COMM%NCPU)
+      INTEGER :: max_
+
+    ! quick return if possible
+      IF (COMM%NCPU == 1) RETURN
+! NO real special handling here (just restore the necessary standard INTEGER);
+! but I doubt that there is still   ANY   T3D alive (and if memory is small)
+      n4=n
+      max_=n4/COMM%NCPU
+      ! do we have sufficient shm workspace to use fast interchange algorithm
+      ! no use conventional M_sumb_d
+      IF (ISHM_CHECK(n4) == 0) THEN
+         CALL M_sumb_d(COMM, vec, n4)
+         RETURN
+      ENDIF
+!----------------------------------------------------------------------
+#else
+!----------------------------------------------------------------------
+      REAL(q), ALLOCATABLE :: vec_inter(:)
+    ! maximum work space for quick sum
+!
+! maximum communication blocks
+! too large blocks are slower on the Pentium architecture
+! probably due to caching
+!
+      INTEGER, PARAMETER :: max_=MPI_BLOCK
+
+    ! quick return if possible
+      IF (COMM%NCPU == 1) RETURN
+
+      IF (n/COMM%NCPU < 2147483648_qi8) THEN
+! here the case that n/COMM%CPU still fits into the 32-bit INTEGER range ...
+         n4=n/COMM%NCPU
+         mmax=MIN(n4,max_)
+      ELSE
+! ... if not: (since max_ is of type 32-bit INTEGER) max_ *must* be the minimum
+         mmax=max_
+      ENDIF
+      ALLOCATE(vec_inter(mmax*COMM%NCPU))
+!----------------------------------------------------------------------
+#endif
+!----------------------------------------------------------------------
+
+      nsummed=0_qi8
+      n_=n/COMM%NCPU
+
+      DO ndo=0_qi8,n_-1_qi8,mmax
+     ! forward exchange
+     ! since the maximum value of mmax is finally determined by max_ (MPI_BLOCK)
+     ! which is a 32-bit INTEGER (and hence can never violate the allowed range
+     ! of 32-bit INTEGER numbers) I could safely assume that "ncount" may be of
+     ! type "INTEGER" -- this was essential since DAXPY and DCOPY as well as
+     ! all MPI calls require 32-bit INTEGER arguments (for a standard BLAS);
+     ! be warned that the situation might change if COMM%NCPU goes into the
+     ! millions (since actually not ncount alone but ncount*COMM%NCPU is handed
+     ! over as argument!) but then not only here but in ALL other subroutines we
+     ! would run into trouble -- or needed BLAS/MPI routines that can accept
+     ! 64-bit INTEGERS (still rare but already existing ...); at the moment I
+     ! could just recommend to set a small enough value for max_ (MPI_BLOCK)
+     ! which must be simply smaller than INT(2147483647/NCPU) ...
+         ncount =MIN(mmax,n_-ndo)
+         nsummed=nsummed+ncount*COMM%NCPU
+
+         CALL M_alltoall_d(COMM, ncount*COMM%NCPU, vec(ndo*COMM%NCPU+1), vec_inter(1))
+        ! sum localy
+         DO i=2, COMM%NCPU
+           CALL DAXPY(ncount, 1.0_q, vec_inter(1+(i-1)*ncount), 1, vec_inter(1), 1)
+         ENDDO
+     ! replicate data (will be send to each proc)
+         DO i=1, COMM%NCPU
+            DO j=1,ncount
+               vec(ndo*COMM%NCPU+j+(i-1)*ncount) = vec_inter(j)
+            ENDDO
+         ENDDO
+     ! backward exchange
+         CALL M_alltoall_d(COMM, ncount*COMM%NCPU, vec(ndo*COMM%NCPU+1), vec_inter(1))
+         CALL DCOPY( ncount*COMM%NCPU, vec_inter(1), 1, vec(ndo*COMM%NCPU+1), 1 )
+      ENDDO
+
+     ! that should be it
+      IF (n_*COMM%NCPU /= nsummed) THEN
+         CALL vtutor%bug("M_sumf_d8: " // str(n_) // " " // str(nsummed), __FILE__, __LINE__)
+      ENDIF
+
+! Now the remaining few elements (we can use M_sumb_d for this task since the
+! number of elements left cannot be larger than "max_" [being a 32-bit INTEGER])
+      IF (n-nsummed /= 0_qi8 ) THEN
+        n4=n-nsummed
+        CALL M_sumb_d(COMM, vec(nsummed+1), n4)
+      ENDIF
+
+#if defined(T3D_SMA)
+     ! nup nothing to do here
+#else
+      DEALLOCATE(vec_inter)
+#endif
+      END SUBROUTINE M_sumf_d8
+
+
+!----------------------------------------------------------------------
+!
 ! M_sumf_s: performs a fast global sum on n singles in
 ! vector vec (algorithm by Kresse Georg)
 !
@@ -4577,6 +4708,60 @@
 
 !----------------------------------------------------------------------
 !
+! M_sum_d8:  performs a sum on n double precision numbers
+! this is for giant arrays with giant dimensions (parameter n potentially
+! exceeding the range of 32-bit INTEGER numbers); this mainly uses special
+! version M_sumf_d_giant of routine M_sumf_d (use of M_sumb_d not always
+! possible for "use_collective_sum" -- only second call always possible)
+!
+!----------------------------------------------------------------------
+
+      SUBROUTINE M_sum_d8(COMM, vec, n)
+      USE mpimy
+      IMPLICIT NONE
+
+      TYPE(communic) COMM
+      INTEGER(qi8) n
+      REAL(q) vec(n)
+! we need additional local variables
+      INTEGER(qi8) max_
+      INTEGER n4
+
+! return ranks that are not in the group
+      IF ( COMM%MPI_COMM == MPI_COMM_NULL ) THEN
+         RETURN
+      ENDIF
+
+#ifdef use_collective_sum
+! this is now a bit more complicated than in routine M_sum_d ...
+      max_=2147483647_qi8          ! 2^31-1 = maximum possible 32-bit integer
+      IF ( n>max_) THEN
+! here we have no choice; we cannot use M_sumb_d (MPI limitations to 32-bit
+! integers!) and therefore we have to use M_sumf_d8 instead ... !!
+         CALL M_sumf_d8(COMM, vec, n)
+      ELSE
+! only "n" smaller than 2^31 (limit for 32-bit integers) allows use of M_sumb_d
+         n4=n
+         CALL M_sumb_d(COMM, vec, n4)
+      ENDIF
+#else
+! ... and also here we need to take a bit care ...
+      max_=MPI_BLOCK
+      IF (n>max_) THEN
+! ... for safety we also have to call the "8" version here ...
+         CALL M_sumf_d8(COMM, vec, n)
+      ELSE
+! ... and since MPI_BLOCK (which is a 32-bit INTEGER) is the possible maximum
+! of n inside this ELSE branch we can keep the call to M_sumb_d here ...
+         n4=n
+         CALL M_sumb_d(COMM, vec, n4)
+      ENDIF
+#endif
+      END SUBROUTINE M_sum_d8
+
+
+!----------------------------------------------------------------------
+!
 ! M_sum_s:  performs a sum on n single prec numbers
 !  it uses either sumb_s or sumf_s
 !
diff -Naur ../original/vasp.6.3.0/src/sphpro.F ./src/sphpro.F
--- ../original/vasp.6.3.0/src/sphpro.F	2022-01-20 16:10:07.000000000 +0100
+++ ./src/sphpro.F	2023-01-04 13:59:43.629249207 +0100
@@ -1461,12 +1461,12 @@
       NIS = NIS+T_INFO%NITYP(NT)
       ENDDO typ
 
-      CALLMPI( M_sum_d( WDES%COMM, ION_SUM, LDIMP*T_INFO%NIONS*WDES%NCDIJ))
-      CALLMPI( M_sum_d( WDES%COMM, ION_SUM_DETAIL, LMDIMP*T_INFO%NIONS*WDES%NCDIJ))
+      CALLMPI( M_sum_d8( WDES%COMM, ION_SUM, 1_qi8*LDIMP*T_INFO%NIONS*WDES%NCDIJ))
+      CALLMPI( M_sum_d8( WDES%COMM, ION_SUM_DETAIL, 1_qi8*LMDIMP*T_INFO%NIONS*WDES%NCDIJ))
       IF (LFINAL) THEN
-        CALLMPI( M_sum_d( WDES%COMM, PAR(1,1,1,1,1), WDES%NB_TOT*WDES%NKPTS*LPAR*T_INFO%NIONP*WDES%NCDIJ))
+        CALLMPI( M_sum_d8( WDES%COMM, PAR(1,1,1,1,1), 1_qi8*WDES%NB_TOT*WDES%NKPTS*LPAR*T_INFO%NIONP*WDES%NCDIJ))
         IF (LORBIT>=12 .AND. LORBIT<=14) THEN
-           CALLMPI( M_sum_d( WDES%COMM, PHAS(1,1,1,1,1), LMDIMP*T_INFO%NIONS*WDES%NKPTS*WDES%NB_TOT*WDES%ISPIN*2))
+           CALLMPI( M_sum_d8( WDES%COMM, PHAS(1,1,1,1,1), 1_qi8*LMDIMP*T_INFO%NIONS*WDES%NKPTS*WDES%NB_TOT*WDES%ISPIN*2))
         ENDIF
       ENDIF
 
diff -Naur ../original/vasp.6.3.0/src/string.F ./src/string.F
--- ../original/vasp.6.3.0/src/string.F	2022-01-20 16:10:07.000000000 +0100
+++ ./src/string.F	2023-01-04 13:59:46.819231533 +0100
@@ -2,7 +2,8 @@
 
 module string
 
-    use prec, only: q,qd
+!jF:  also "qi8" (64-bit integer data type) is needed for my modifications below
+    use prec, only: q,qd,qi8
     USE base, ONLY:  TOREAL
 
     implicit none
@@ -30,9 +31,10 @@
     !! is used instead of the given format whenever the format would result in
     !! an error or an overflow.
     !!
+!jF:  added here "integer8Format" for handling 64-bit integers ...
     interface str
         module procedure logicalDefault, logicalCompact, logicalFormat, &
-            integerFormat, integerArray, realDefault, realFormat, &
+            integerFormat, integer8Format, integerArray, realDefault, realFormat, &
             realArray, real2dArray, complexDefault, complexFormat, &
             complexArray, complex2dArray
 #if !defined(noQuadPrecision) || defined(qd_emulate)
@@ -296,6 +298,21 @@
         res = trim(tmp)
     end function integerFormat
 
+ !jF:  added here "integer8Format" for handling 64-bit integer numbers ...
+    pure function integer8Format(ivar8, formatString) result (res)
+        integer(qi8), intent(in) :: ivar8
+        character(len=*), intent(in), optional :: formatString
+        character(len=*), parameter :: defaultFormat = '(i0)'
+        character(len=:), allocatable :: res
+        character(len=maxLength) tmp
+        integer ierr
+        if (present(formatString)) &
+            write(tmp, formatString, iostat=ierr) ivar8
+        if (.not.present(formatString).or.ierr /= 0.or.tmp(1:1) == '*') &
+            write(tmp, defaultFormat) ivar8
+        res = trim(tmp)
+    end function integer8Format
+
     pure function integerArray(iarr, formatString) result (res)
         integer, intent(in) :: iarr(:)
         character(len=*), intent(in), optional :: formatString

dongshen_wen
Newbie
Newbie
Posts: 8
Joined: Wed Nov 02, 2022 10:30 am

Re: internal error in mpi.F

#5 Post by dongshen_wen » Fri Jan 06, 2023 11:56 am

Hi,
I was using 6.3.2 vasp_ncl and saw similar errors. Does this patch also work for 6.3.2? It's a different subroutine M_sumb_d though.

Thanks,
Dongsheng

-----------------------------------------------------------------------------
| _ ____ _ _ _____ _ |
| | | | _ \ | | | | / ____| | | |
| | | | |_) | | | | | | | __ | | |
| |_| | _ < | | | | | | |_ | |_| |
| _ | |_) | | |__| | | |__| | _ |
| (_) |____/ \____/ \_____| (_) |
| |
| internal error in: mpi.F at line: 1359 |
| |
| M_sumb_d: invalid vector size n -1953955840 |
| |
-----------------------------------------------------------------------------

alexey.tal
Global Moderator
Global Moderator
Posts: 320
Joined: Mon Sep 13, 2021 12:45 pm

Re: internal error in mpi.F

#6 Post by alexey.tal » Mon Jan 09, 2023 2:45 pm

Dear Dongsheng,

If the error occurs in subroutine SPHPRO_FAST, this patch should fix it for VASP 6.3.2. Otherwise, you should create a new thread and attach all the files, so that we can find what the problem is.

dongshen_wen
Newbie
Newbie
Posts: 8
Joined: Wed Nov 02, 2022 10:30 am

Re: internal error in mpi.F

#7 Post by dongshen_wen » Tue Jan 10, 2023 10:07 am

Dear Alexey,

I tried the patch above but it did not seem to work. I still got the same error message. It seems to me it came from the M_sumb_d subroutine on line 1329 in mpi.F. I saw that the patch updates the M_sum_d to M_sum_d8 so I'm not sure if this is the case.
I've also tried commenting out the LORBIT tag, the error persisted. So I guess it might not come from the same subroutine SPHPRO_FAST as you mentioned.

I will open a new thread to update the files.

Best,
Dongsheng
alexey.tal wrote: Mon Jan 09, 2023 2:45 pm Dear Dongsheng,

If the error occurs in subroutine SPHPRO_FAST, this patch should fix it for VASP 6.3.2. Otherwise, you should create a new thread and attach all the files, so that we can find what the problem is.

andreas.singraber
Global Moderator
Global Moderator
Posts: 249
Joined: Mon Apr 26, 2021 7:40 am

Re: internal error in mpi.F

#8 Post by andreas.singraber » Thu Jan 12, 2023 9:20 am

Hello Dongsheng,

as you seem to face a different problem we will continue discussing in this thread: https://vasp.at/forum/viewtopic.php?f=3&t=18793

Best,
Andreas Singraber

Post Reply