1

我一直在调试模式下(-O0 -g -traceback -fpe3打开标志)使用英特尔编译器版本 13.1.3.192 运行巨大的 Fortran 代码。它给了我以下输出消息:

... ...
forrtl: warning (402): fort: (1): In call to MPI_ALLGATHER, an array temporary was created for argument #1

forrtl: error (65): floating invalid
Image              PC                Routine            Line        Source
arts               00000000016521D9  pentadiagonal_ser         385  pentadiagonal.f90
arts               0000000001644862  pentadiagonal_             62  pentadiagonal.f90
arts               00000000004DF167  implicit_solve_x_        1201  implicit.f90
arts               0000000000538079  scalar_bquick_inv         383  scalar_bquick.f90
arts               00000000004EFEAC  scalar_step_              190  scalar.f90
arts               0000000000401744  simulation_run_           182  simulation.f90
arts               0000000000401271  MAIN__                     10  main.f90
arts               0000000000400FAC  Unknown               Unknown  Unknown
arts               000000000420E444  Unknown               Unknown  Unknown
arts               0000000000400E81  Unknown               Unknown  Unknown

错误的来源来自子程序 pentadiagonal_serial,它是求解一个五对角矩阵:

subroutine pentadiagonal_serial(A,B,C,D,E,R,n,lot)
  use precision
  implicit none

  integer, intent(in) :: n,lot
  real(WP), dimension(lot,n) :: A     ! LOWER-2
  real(WP), dimension(lot,n) :: B     ! LOWER-1
  real(WP), dimension(lot,n) :: C     ! DIAGONAL
  real(WP), dimension(lot,n) :: D     ! UPPER+1
  real(WP), dimension(lot,n) :: E     ! UPPER+2
  real(WP), dimension(lot,n) :: R     ! RHS - RESULT
  real(WP), dimension(lot) :: const
  integer :: i

  if (n .eq. 1) then
    ! Solve 1x1 system
    R(:,1) = R(:,1)/C(:,1)
    return
  else if (n .eq. 2) then
    ! Solve 2x2 system
    const(:) = B(:,2)/C(:,1)
    C(:,2) = C(:,2) - D(:,1)*const(:)
    R(:,2) = R(:,2) - R(:,1)*const(:)
    R(:,2) = R(:,2)/C(:,2)
    R(:,1) = (R(:,1) - D(:,1)*R(:,2))/C(:,1)
    return
 end if

 ! Forward elimination
 do i=1,n-2
     ! Eliminate A(2,i+1)
     const(:) = B(:,i+1)/(C(:,i)+tiny(1.0_WP))
     C(:,i+1) = C(:,i+1) - D(:,i)*const(:)
     D(:,i+1) = D(:,i+1) - E(:,i)*const(:)
     R(:,i+1) = R(:,i+1) - R(:,i)*const(:)

其中行

  const(:) = B(:,i+1)/(C(:,i)+tiny(1.0_WP))

导致错误。我试图打印出的值const(:)并发现确实存在Infinity值。但是,我不明白为什么它会产生无穷大。据我所见,为了避免分母为零,tiny(1.0_WP)添加了C(:,i),现在分母几乎不可能为零......我还检查了何时调用此子例程,所有内容都被初始化或在声明后赋予一个值。所以我想不出哪里会出错。

4

1 回答 1

1

(评论中的答案。请参阅没有答案的问题,但在评论中解决了问题(或在聊天中扩展)。评论中有很多聊天,很难提取实际答案,但 OP 表示已解决. )

@SethMMorton 写道:

如果 的任何值C在附近,-tiny那么您仍然可以在分母中以零结束。

您是否打印出 的值C?小值是否与 中的无穷大相关const

@Alexander Vogt 写道:

好吧,向无穷大添加一些东西会导致浮点异常......而且我不确定 ifort 是否可以检测到被无穷大除为零,我想这是另一个浮点异常!您最好确保 C 内部没有无限值。

@Stefan 写道:

解决方法:?gbsvMKL/Lapack 的例程怎么样?然后,最后一个提示:尝试使用gfortran. 不同的编译器会创建不同的编译时/运行时警告和/或错误,并且可能gfortran会更早地检测到您的问题。

于 2015-01-25T17:27:00.363 回答