未定义的符号编译fortran 66/77在c ++到八度?

时间:2014-05-12 13:49:54

标签: c++ fortran octave

当我在ising3d_fort2oct.cc中执行ising3d.f时,我一直收到“未定义的符号:ising3d_”错误。任何人都可以提供有关我可以更改ising3d.f或ising3d_fort2oct.cc以解决此问题的建议吗?

Octave终端命令:

octave:12> system("gfortran ising3d.f -o ising3d")
ans = 0
octave:13> mkoctfile ising3d_fort2oct.cc
octave:14> ising3d_fort2oct
error: /home/b/Desktop/Umaine_Classes/CHY_573/3disingmodelf/ising3d_fort2oct.oct: failed to load: /home/b/Desktop/Umaine_Classes/CHY_573/3disingmodelf/ising3d_fort2oct.oct: undefined symbol: ising3d_
octave:14> 

ising3d_fort2oct.cc:动态链接C ++函数调用fortran子程序ising3d.f:

#include <octave/oct.h>
#include <octave/f77-fcn.h>
extern "C" void F77_FUNC (ising3d,ISING3D)();
DEFUN_DLD (ising3d_fort2oct, args , ,"work in progress")
    {
        octave_value_list retval;
        F77_FUNC(ising3d,ISING3D)();
        return retval;
    }

ising3d.f:真正古老的fortran子程序:

          SUBROUTINE ising3d()
C         3D ISING MODEL
C         Critical temperature TC = 4.5116=1/.22165
          DIMENSION IS(10,10,10), EX(13)
          REAL *8 R(1)
          DATA IS/1000*1/
          ITMAX=5000
          ISTART=4000
          L=10
          NR=1
          ISEED=768521034
          M=L*L*L
          DO 1000 K=1,24
          TR=0.05+(K-1)*.05
          T=TR/.221655
          MR=0.
          DO 3 I=1, 13, 2
3         EX(I)=EXP(-2*(I-7.)/T)
          DO 2 ITIME=1,ITMAX
          DO 1 K1=1,L
          K1P1=K1+1
          K1M1=K1-1
          IF(K1.EQ.1) K1M1=L
          IF(K1.EQ.L) K1P1=1
          DO 1 K2=1,L
          K2P1=K2+1
          K2M1=K2-1
          IF(K2.EQ.1) K2M1=L
          IF(K2.EQ.L) K2P1=1
          DO 1 K3=1,L
          K3P1=K3+1
          K3M1=K3-1
          IF(K3.EQ.1) K3M1=L
          IF(K3.EQ.L) K3P1=1
        IEN=7+IS(K3,K2,K1)*(IS(K3M1,K2,K1)+IS(K3P1,K2,K1)+IS(K3,K2M1,K1)
     &  +IS(K3,K2P1,K1)+IS(K3,K2,K1M1)+IS(K3,K2,K1P1))
          CALL GGUBS(ISEED,NR,R)
          IF(EX(IEN).LT.R(1)) GOTO 1
          IS(K3,K2,K1)=-IS(K3,K2,K1)
          M=M+2*IS(K3,K2,K1)
C          WRITE(*,*) M,ITIME
1         CONTINUE
          IF (ITIME.GT.ISTART) MR=MR+M
2         CONTINUE
          WRITE(*,*) FLOAT(MR)/1000./FLOAT((ITMAX-ISTART)),TR
          WRITE(1,*) FLOAT(MR)/1000./FLOAT((ITMAX-ISTART)),TR
1000      CONTINUE
          STOP
          END
          RETURN
          END

        SUBROUTINE GGUBS(ISEED,NR,R)
        IMPLICIT REAL *8(A-H,O-Z)
        DIMENSION R(NR)
        DATA D2P31M/2147483647.D0/
        DATA D2P31/2147483648.D0/
        DO 7 I=1,NR
        ISEED=MOD(16807.D0*ISEED, D2P31M)
        R(I)=ISEED/D2P31
    7   CONTINUE
        RETURN
        END

1 个答案:

答案 0 :(得分:1)

问题不在于代码,而是在编译和链接要从Octave调用的oct文件的方式。构建ising3d_fort2oct时,您没有指示它与ising3d Fortran函数链接或如何找到它。

您可以通过两种方式之一将这两个源文件链接在一起。您可以单独编译每个文件,然后将生成的目标文件链接在一起,或者您可以使用两个源文件调用mkoctfile一次,然后让它为您完成所有操作。

要使用一个命令从两个源构建oct文件,请执行

octave:1> mkoctfile ising3d_fort2oct.cc ising3d.f

在Octave shell中,其中mkoctfile被告知第二个源文件要编译并与第一个源文件链接在一起。

要单独构建Fortran子例程,就像您尝试的那样,然后稍后将其与oct文件包装器链接,您可以改为

octave:1> system ("gfortran -c ising3d.f -o ising3d.o");
octave:2> mkoctfile ising3d_fort2oct.cc ising3d.o

请注意,您仍然需要显式链接oct文件和目标文件。否则,将编译oct文件,假设该函数由Octave本身或系统库提供,这就是为什么你得到一个未定义的符号错误。